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 к отображениям по умолчанию.
Другие вопросы по тегам