Java toString() 使用反射?
-
22-07-2019 - |
题
前几天,我正在为 Java 中的一个类编写 toString(),方法是手动将类的每个元素写入字符串,我突然想到,使用反射可能可以创建一个可以工作的通用 toString() 方法所有课程。IE。它会计算出字段名称和值并将它们发送到字符串。
获取字段名称非常简单,这是同事的想法:
public static List initFieldArray(String className) throws ClassNotFoundException {
Class c = Class.forName(className);
Field field[] = c.getFields();
List<String> classFields = new ArrayList(field.length);
for (int i = 0; i < field.length; i++) {
String cf = field[i].toString();
classFields.add(cf.substring(cf.lastIndexOf(".") + 1));
}
return classFields;
}
使用工厂,我可以通过在第一次调用 toString() 时存储一次字段来降低性能开销。然而,找到这些值的成本可能要高得多。
由于反射的性能,这可能更假设,而不是实际。但我对反射的想法以及如何使用它来改进我的日常编程感兴趣。
解决方案
Apache 公共语言 反射到字符串生成器 为你做这个。
import org.apache.commons.lang3.builder.ReflectionToStringBuilder
// your code goes here
public String toString() {
return ReflectionToStringBuilder.toString(this);
}
其他提示
另一种选择,如果您没有问题JSON,是谷歌的GSON库。
public String toString() {
return new GsonBuilder().setPrettyPrinting().create().toJson(this);
}
这将做反射给你。这产生了很好的,易于阅读JSON文件。易于阅读是相对的,非高科技人可能会发现JSON恐吓。
您可以使GSONBuilder一个成员变量太多,如果你不希望新的它,每一次。
如果您有不能打印(如流),或者你只是不想打印数据的数据,你可以添加@expose标签要打印,然后使用下面的行的属性。
new GsonBuilder()
.setPrettyPrinting()
.excludeFieldsWithoutExposeAnnotation()
.create()
.toJson(this);
W /反射,因为我没有意识到apache的库:
(注意,如果你这样做,你可能需要处理的子对象,并确保他们正确打印 - 特别是,数组不会告诉你任何有用的)
@Override
public String toString()
{
StringBuilder b = new StringBuilder("[");
for (Field f : getClass().getFields())
{
if (!isStaticField(f))
{
try
{
b.append(f.getName() + "=" + f.get(this) + " ");
} catch (IllegalAccessException e)
{
// pass, don't print
}
}
}
b.append(']');
return b.toString();
}
private boolean isStaticField(Field f)
{
return Modifier.isStatic(f.getModifiers());
}
如果您正在使用Eclipse,你也可以看看 JUtils的toString发电机,该做它静态(产生在源代码中的方法)。
不是反射,但我有一看生成toString方法(与equals /的hashCode一起)作为使用字节码操作的汇编后步骤。 结果进行混合。
您可以使用已经实现的库,例如 反射到字符串生成器 来自 Apache commons-lang。正如前面提到的。
或者自己写类似的smt 反射API.
这是 一些例子:
class UniversalAnalyzer {
private ArrayList<Object> visited = new ArrayList<Object>();
/**
* Converts an object to a string representation that lists all fields.
* @param obj an object
* @return a string with the object's class name and all field names and
* values
*/
public String toString(Object obj) {
if (obj == null) return "null";
if (visited.contains(obj)) return "...";
visited.add(obj);
Class cl = obj.getClass();
if (cl == String.class) return (String) obj;
if (cl.isArray()) {
String r = cl.getComponentType() + "[]{";
for (int i = 0; i < Array.getLength(obj); i++) {
if (i > 0) r += ",";
Object val = Array.get(obj, i);
if (cl.getComponentType().isPrimitive()) r += val;
else r += toString(val);
}
return r + "}";
}
String r = cl.getName();
// inspect the fields of this class and all superclasses
do {
r += "[";
Field[] fields = cl.getDeclaredFields();
AccessibleObject.setAccessible(fields, true);
// get the names and values of all fields
for (Field f : fields) {
if (!Modifier.isStatic(f.getModifiers())) {
if (!r.endsWith("[")) r += ",";
r += f.getName() + "=";
try {
Class t = f.getType();
Object val = f.get(obj);
if (t.isPrimitive()) r += val;
else r += toString(val);
} catch (Exception e) {
e.printStackTrace();
}
}
}
r += "]";
cl = cl.getSuperclass();
} while (cl != null);
return r;
}
}
下面是Netbeans的相当于Olivier的答案; 智能代码生成插件的Netbeans 。