Dozer String для отображения enum
У меня есть такой enum:
public enum PartnershipIndicator {
VENDOR("VENDOR"), COPARTNER("COPARTNER"), BUYER("BUYER");
String code;
private PartnershipIndicator(String code) {
this.code = code;
}
public String getCode() {
return code;
}
public static PartnershipIndicator valueOfCode(String code) {
for (PartnershipIndicator status : values()) {
if (status.getCode().equals(code)) {
return status;
}
}
throw new IllegalArgumentException(
"Partnership status cannot be resolved for code " + code);
}
@Override
public String toString() {
return code;
}
}
Мне нужно преобразовать его в строку и наоборот. Теперь это делается с помощью пользовательского конвертера. Но я хочу сделать это через сопоставления бульдозера (если это возможно). Если я не пишу никаких сопоставлений в бульдозерную конфинцию, я получаю
org.dozer.MappingException: java.lang.NoSuchMethodException: by.dev.madhead.demo.test_java.model.PartnershipIndicator.<init>()
исключение. Я не могу добавить открытый конструктор по умолчанию в enum, так как это невозможно. Итак, я написал трюк с внутренним кодом и valueOfCode() / toString(). Это не работает. Затем я сопоставил его в конфигурации Dozer:
<mapping>
<class-a>java.lang.String</class-a>
<class-b create-method="valueOfCode">by.dev.madhead.demo.test_java.model.PartnershipIndicator</class-b>
</mapping>
Это не работает. Я пробовал valueOfCode(), односторонние сопоставления. Ничего не работает Перечисление enum в String тоже не работает, я получаю пустые строки. Есть идеи?
2 ответа
Не уверен, что это все еще проблема, но, возможно, поможет всем, кто ищет. Но здесь реализовано решение этого:
@Override
public Object convert(Object destination, Object source, Class<?> destinationClass, Class<?> sourceClass) {
if(source == null)
return null;
if(destinationClass != null){
if(destinationClass.getSimpleName().equalsIgnoreCase("String")){
return this.getString(source);
}else if( destinationClass.isEnum()){
return this.getEnum(destinationClass, source);
}else{
throw new MappingException(new StrBuilder("Converter ").append(this.getClass().getSimpleName())
.append(" was used incorrectly. Arguments were: ")
.append(destinationClass.getClass().getName())
.append(" and ")
.append(source).toString());
}
}
return null;
}
private Object getString(Object object){
String value = object.toString();
return value;
}
private Object getEnum(Class<?> destinationClass, Object source){
Object enumeration = null;
Method [] ms = destinationClass.getMethods();
for(Method m : ms){
if(m.getName().equalsIgnoreCase("valueOf")){
try {
enumeration = m.invoke( destinationClass.getClass(), (String)source);
}
catch (IllegalArgumentException e) {
e.printStackTrace();
}
catch (IllegalAccessException e) {
e.printStackTrace();
}
catch (InvocationTargetException e) {
e.printStackTrace();
}
return enumeration;
}
}
return null;
}
Класс StrBuilder при создании сообщения об исключении взят из библиотек apache common-lang. Но кроме этого простое отражение, чтобы решить эту проблему. Просто добавьте в класс, который реализует CustomConverter, а затем в свой xml-файл dozer mapping добавьте следующую конфигурацию:
<configuration>
<custom-converters>
<converter type="com.yourcompany.manager.utils.dozer.converters.EnumStringBiDirectionalDozerConverter">
<class-a>java.lang.Enum</class-a>
<class-b>java.lang.String</class-b>
</converter>
</custom-converters>
</configuration>
Обратите внимание, что вы можете перечислить конфигурацию только один раз между всеми вашими файлами сопоставления (если у вас их несколько), в противном случае dozer будет жаловаться. Что я обычно делаю, это помещаю мои собственные конфигурации конвертера в один файл для простоты. Надеюсь это поможет!
В Dozer нет перечисления по умолчанию для сопоставления строк. См. Преобразование типов данных из Dozer-документов. Итак, у вас есть два варианта:
- Вы можете написать собственный конвертер, который использует дженерики для обработки любого перечисления.
- Или вы можете отправить патч в Dozer, чтобы добавить enum<->String mapping к отображениям по умолчанию.