web-dev-qa-db-ja.com

Java:リフレクションを使用して列挙型をインスタンス化する

次のようなテキストファイルがあるとします。

my_setting = ON
some_method = METHOD_A
verbosity = DEBUG
...

それに応じて、対応するオブジェクトを更新したい:

Setting my_setting = ON;
Method some_method = METHOD_A;
Verbosity verbosity = DEBUG;
...

すべてが異なる種類の列挙型である場合。

列挙値をインスタンス化する一般的な方法が欲しいです。つまり、実行時にリフレクションを使用し、事前にオブジェクトの列挙型を知らない。

私はこのような何かを想像していたでしょう:

for (ConfigLine line : lines)
{
   String[] tokens = line.string.split("=", 2);
   String name = tokens[0].trim();
   String value = tokens[1].trim();

   try
   {
      Field field = this.getClass().getDeclaredField(name);   
      if(field.getType().isEnum())
      {
         // doesn't work (cannot convert String to enum)
         field.set(this, value);
         // invalid code (some strange generics issue)
         field.set(this, Enum.valueOf(field.getType().getClass(), value));
      }
      else
      { /*...*/ }
   }
   catch //...
}

問題は、代わりに何が必要かということです。文字列表現が与えられた未知の列挙をインスタンス化することさえ可能ですか?

44
dagnelies
_field.set(this, Enum.valueOf((Class<Enum>) field.getType(), value));
_
  • getClass()getType()の後に呼び出されるべきではありません-Classインスタンスのクラスを返します
  • Classenumであることを既に知っているため、一般的な問題を回避するために_Class<Enum>_をキャストできます。
93
Bozho

キャストなしの代替ソリューション

try {
    Method valueOf = field.getType().getMethod("valueOf", String.class);
    Object value = valueOf.invoke(null, param);
    field.set(test, value);
} catch ( ReflectiveOperationException e) {
    // handle error here
}
8
Rebzie

余分なgetClass呼び出しがあり、キャストする必要があります(Bozhoごとのより具体的なキャスト):

field.set(test, Enum.valueOf((Class<Enum>) field.getType(), value));
4

これと同様のEnumをコーディングできます。

public enum Setting {

    ON("ON"),OFF("OFF");

    private final String setting;

    private static final Map<String, Setting> stringToEnum = new ConcurrentHashMap<String, Setting>();
    static {
        for (Setting set: values()){
            stringToEnum.put(set.setting, set);
        }
    }

    private Setting(String setting) {
        this.setting = setting;
    }

    public String toString(){
        return this.setting;
    }

    public static RequestStatus fromString(String setting){
        return stringToEnum.get(setting);
    }   
}

その後、リフレクションなしでStringからEnumを簡単に作成できます。

Setting my_settings = Setting.fromString("ON");

この解決策は私自身のものではありません。私は他のどこかからそれを読みましたが、ソースを思い出すことができません。

0
ThiamTeck

受け入れられた回答は、Enum <T extends Enum <T >>の代わりに生のタイプEnumを使用するため、警告になります。

これを回避するには、次のような汎用メソッドを使用する必要があります。

@SuppressWarnings("unchecked")
private <T extends Enum<T>> T createEnumInstance(String name, Type type) {
  return Enum.valueOf((Class<T>) type, name);
}

次のように呼び出します。

Enum<?> enum = createEnumInstance(name, field.getType());
field.set(this, enum);
0
Dev Vercer