57

前几天我正在为 Java 中的一个类编写一个 toString(),方法是将类的每个元素手动写入一个 String,我突然想到,使用反射可以创建一个可以工作的通用 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() 时存储一次字段来降低性能开销。然而,找到这些值可能要昂贵得多。

由于反射的性能,这可能比实际更假设。但我对反射的想法以及如何使用它来改进我的日常编程很感兴趣。

4

7 回答 7

101

Apache commons-lang ReflectionToStringBuilder为您执行此操作。

import org.apache.commons.lang3.builder.ReflectionToStringBuilder

// your code goes here

public String toString() {
   return ReflectionToStringBuilder.toString(this);
}
于 2009-01-14T16:53:44.843 回答
10

如果您对 JSON 没问题,另一种选择是 Google 的 GSON 库。

public String toString() {
    return new GsonBuilder().setPrettyPrinting().create().toJson(this);
}

它会为你做反射。这会生成一个漂亮、易于阅读的 JSON 文件。相对易于阅读,非技术人员可能会觉得 JSON 令人生畏。

如果您不想每次都更新它,您也可以将 GSONBuilder 设为成员变量。

如果您有无法打印的数据(如流)或您不想打印的数据,您可以将@Expose 标签添加到您想要打印的属性,然后使用以下行。

 new GsonBuilder()
.setPrettyPrinting()
.excludeFieldsWithoutExposeAnnotation()
.create()
.toJson(this);
于 2016-01-19T17:28:19.157 回答
6

W/reflection,因为我不知道 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());
}
于 2009-01-14T18:23:17.223 回答
5

如果您使用的是 Eclipse,您还可以查看JUtils toString generator,它是静态执行的(在源代码中生成方法)。

于 2009-01-14T17:00:42.117 回答
4

不是反射,但我查看了生成 toString 方法(以及 equals/hashCode)作为使用字节码操作的编译后步骤。结果喜忧参半。

于 2009-01-14T18:17:35.113 回答
4

您可以使用已经实现的库,例如来自 Apache commons-lang的ReflectionToStringBuilder 。如前所述。

或者自己用反射 API编写类似的 smt 。

这是一些例子

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;
   }    
}
于 2014-09-20T22:16:46.957 回答
2

这是相当于 Olivier 答案的 Netbeans;用于 Netbeans 的 smart-codegen 插件

于 2009-03-17T21:59:44.473 回答