Java:使用反射实例化枚举

时间:2010-09-17 13:56:22

标签: java reflection enums

假设您有一个文本文件,如:

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 //...
}

问题是:应该改为什么?是否可以在给定String表示的情况下实例化一个未知的枚举?

5 个答案:

答案 0 :(得分:91)

field.set(this, Enum.valueOf((Class<Enum>) field.getType(), value));
    不应在getClass()之后
  • getType() - 它返回Class实例的类
  • 您可以投放Class<Enum>,以避免一般性问题,因为您已经知道Classenum

答案 1 :(得分:7)

没有投射的替代解决方案

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
}

答案 2 :(得分:4)

你有一个额外的getClass电话,你必须施放(每个Bozho更具体的演员阵容):

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

答案 3 :(得分:0)

您可以使用类似的代码编写您的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);
    }   
}

然后你可以轻松地从字符串创建Enum而不用反射:

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

此解决方案并非来自我自己。我是从其他地方读到的,但我记不起来了。

答案 4 :(得分:0)

接受的答案会产生警告,因为它使用原始类型Enum而不是Enum&lt; T extends Enum&lt; T&gt;&gt ;.

要解决此问题,您需要使用这样的通用方法:

@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);