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 //...
}
Вопрос в том, что должно быть вместо этого? Возможно ли даже создать экземпляр неизвестного перечисления с его строковым представлением?
Ответы
Ответ 1
field.set(this, Enum.valueOf((Class<Enum>) field.getType(), value));
-
getClass()
после getType()
не следует вызывать - он возвращает класс экземпляра Class
- Вы можете использовать
Class<Enum>
, чтобы избежать общих проблем, потому что вы уже знаете, что Class
является enum
Ответ 2
У вас есть дополнительный вызов getClass
, и вам нужно сделать бросок (более конкретный бросок на Божо):
field.set(test, Enum.valueOf((Class<Enum>) field.getType(), value));
Ответ 3
Альтернативное решение без литья
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
}
Ответ 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);
}
}
Тогда вы можете легко создать Enum из String без отражения:
Setting my_settings = Setting.fromString("ON");
Это решение исходит не от меня. Я читал его откуда-то еще, но я не могу вспомнить источник.
Ответ 5
Принятый ответ приводит к предупреждениям, потому что он использует необработанный тип Enum вместо Enum < T extends Enum <T → .
Чтобы обойти это, вам нужно использовать общий метод, подобный этому:
@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);