web-dev-qa-db-ja.com

Java toString()リフレクションを使用していますか?

先日、クラスのtoString()をJavaでクラスの各要素を文字列に手動で書き出すことで書き込んでいましたが、リフレクションを使用して作成できる可能性があることに気付きましたすべてのクラスで機能する一般的なtoString()メソッドIEはフィールド名と値を把握し、それらをStringに送信します。

フィールド名の取得は非常に簡単です。同僚が思いついたものを次に示します。

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()が呼び出されたときにフィールドを1回保管することで、パフォーマンスのオーバーヘッドを削減できます。ただし、値を見つけることははるかに高価になる可能性があります。

リフレクションのパフォーマンスにより、これはより仮説的であり、実際的である可能性があります。しかし、私はリフレクションのアイデアと、それを使用して日常のプログラミングを改善する方法に興味があります。

50
James McMahon

Apache commons-lang ReflectionToStringBuilder がこれを行います。

import org.Apache.commons.lang3.builder.ReflectionToStringBuilder

// your code goes here

public String toString() {
   return ReflectionToStringBuilder.toString(this);
}
85
krosenvold

JSONに問題がない場合の別のオプションは、GoogleのGSONライブラリです。

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

それはあなたのために反射をするつもりです。これにより、読みやすく、読みやすいJSONファイルが生成されます。読みやすいものは相対的であり、技術者でない人はJSONを威圧的に感じるかもしれません。

毎回新しいものにしたくない場合は、GSONBuilderもメンバー変数にすることができます。

印刷できないデータ(ストリームなど)がある場合、または印刷したくないデータがある場合は、@ Exposeタグを印刷する属性に追加して、次の行を使用できます。

 new GsonBuilder()
.setPrettyPrinting()
.excludeFieldsWithoutExposeAnnotation()
.create()
.toJson(this);
7
Daniel Schmidt

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());
}
6
Steve B.

Eclipseを使用している場合は、静的に実行する(ソースコードでメソッドを生成する) JUtils toString generator も確認できます。

5
Olivier

Apache commons-langの ReflectionToStringBuilder として、すでに実装されているライブラリを使用できます。述べたように。

または、reflection 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;
   }    
}
4
nazar_art

リフレクションではありませんが、バイトコード操作を使用したコンパイル後のステップとして、toStringメソッド(equals/hashCodeと共に)を生成することを検討しました。 結果は混合されました。

4
McDowell

これは、Olivierの回答に相当するNetbeansです。 Netbeansのsmart-codegenプラグイン

2
James McMahon