Как десериализовать Enum с помощью Джексона? - PullRequest
2 голосов
/ 20 июня 2020

Пользователь отправляет {"lang": ["en_US", "en_UK"]} POST в REST API. Я хочу заполнить объект, имеющий это свойство List<EnmLanguage> lang;. Все остальное работает, кроме JSON массивов.

My EnmLanguage. java это:

@JsonSerialize(
    using = BaseEnumSerializer.class
)
@JsonDeserialize(
    using = BaseEnumDeserializer.class
)
public enum EnmLanguage implements BaseEnum, Serializable {
    en_US {
        public String getCode() {
            return "en_US";
        }

        public String getText(EnmLanguage s) {
            return s == EnmLanguage.en_US ? "English" : "İngilizce";
        }
    },

My BaseEnum. java is:

public interface BaseEnum {
    Object getCode();

    String getText(EnmLanguage lang);

    default String getText() {
        return this.getText(EnmLanguage.tr_TR);
    }

    default Map<String, Object> getMap(EnmLanguage lang) {
        return new HashMap<String, Object>() {
            {
                this.put("name", BaseEnum.this.getText(lang));
                this.put("id", BaseEnum.this.getCode());
            }
        };
    }

    default Map<String, Object> getMap() {
        return this.getMap(EnmLanguage.tr_TR);
    }

    static <T extends Enum<T> & BaseEnum> T fromCode(Class<T> parent, Object code) {
        Map<Object, T> lookup = new HashMap();
        Enum[] var3 = (Enum[])parent.getEnumConstants();
        int var4 = var3.length;

        for(int var5 = 0; var5 < var4; ++var5) {
            T d = var3[var5];
            lookup.put(((BaseEnum)d).getCode().toString(), d);
        }

        return (Enum)lookup.get(code.toString());
    }
}

Мой BaseEnumDeserialiazer:

public class BaseEnumDeserializer<T extends Enum<T> & BaseEnum> extends StdDeserializer<T> {
    public BaseEnumDeserializer() {
        this((Class)null);
    }

    public BaseEnumDeserializer(Class<T> vc) {
        super(vc);
    }

    public T deserialize(JsonParser jsonparser, DeserializationContext context) throws IOException {
        Field field = this.findField(jsonparser.getCurrentName(), jsonparser.getCurrentValue().getClass());
        Class<T> javaType = field.getType();
        return BaseEnum.fromCode(javaType, jsonparser.getText());
    }

    public Field findField(String name, Class<?> c) {
        while(c != null) {
            Field[] var3 = c.getDeclaredFields();
            int var4 = var3.length;

            for(int var5 = 0; var5 < var4; ++var5) {
                Field field = var3[var5];
                if (!Modifier.isStatic(field.getModifiers()) && field.getName().equals(name)) {
                    return field;
                }
            }

            c = c.getSuperclass();
        }

        return null;
    }
}

Но в конце он не преобразует "lang": ["en_US", "en_UK"] в List из EnmLanguage. Я думаю, он не знает, в какой BaseEnum конвертируется. Как мне сообщить ему, что он должен преобразовать его в List<EnmLanguage>?

Ответы [ 2 ]

2 голосов
/ 29 июня 2020

Для вашего случая c, вероятно, будет достаточно добавить метод stati c @JsonCreator внутри базового типа.

interface Lang {
    @JsonCreator
    static Lang valueOf(String value) {
        return Languages.valueOf(value);
    }
}

@JsonFormat(shape = JsonFormat.Shape.STRING)
enum Languages implements Lang {
    en_US,
    en_GB
}

И затем вы можете проанализировать его как любой другой класс

class LangTest {
    private final ObjectMapper mapper = new ObjectMapper()
            .setAnnotationIntrospector(new JacksonAnnotationIntrospector());

    @Test
    void readsJson() throws IOException {
        final Lang[] tree = this.mapper
                .readerFor(Lang[].class)
                .readValue("[\"en_US\", \"en_GB\"]");
        Assertions.assertArrayEquals(
                tree,
                new Lang[]{
                        Languages.en_US,
                        Languages.en_GB
                }
        );
    }
}
1 голос
/ 30 июня 2020

Вот простой BaseEnum интерфейс,

public interface BaseEnum {
    Object getCode();

    String getText();
}

Вот EnmLanguage, реализующий BaseEnum интерфейс,

public enum EnmLanguage implements BaseEnum {

    en_US("en_US", "English"),
    en_UK("en_UK", "English"),
    tr_TR("tr_TR", "Turkish");

    private String code;

    private String text;

    EnmLanguage(String code, String text) {
        this.code = code;
        this.text = text;
    }

    @Override
    public String getCode() {
        return code;
    }

    @Override
    public String getText() {
        return text;
    }
}

Вот еще EnmElement, реализующий BaseEnum interface,

public enum EnmElement implements BaseEnum {

    H("H", "Hydrogen"),
    He("He", "Helium"),
    Li("Li", "Lithium");

    private String code;

    private String text;

    EnmElement(String code, String text) {
        this.code = code;
        this.text = text;
    }

    @Override
    public String getCode() {
        return code;
    }

    @Override
    public String getText() {
        return text;
    }
}

Вот простой модульный тест,

import com.fasterxml.jackson.databind.ObjectMapper;
import org.junit.Test;

import java.io.IOException;

import static org.junit.Assert.assertArrayEquals;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;

public class EnumTest {

    @Test
    public void testDeserializeLanguage() throws IOException {
        ObjectMapper mapper = new ObjectMapper();

        EnmLanguage[] expected = {EnmLanguage.en_US, EnmLanguage.tr_TR};
        EnmLanguage[] actual = mapper
                .readerFor(EnmLanguage[].class)
                .readValue(mapper.writeValueAsString(expected));
        assertNotNull(actual);
        assertArrayEquals(expected, actual);
    }

    @Test
    public void testDeserializeElement() throws IOException {
        ObjectMapper mapper = new ObjectMapper();

        EnmElement[] expected = {EnmElement.H, EnmElement.He, EnmElement.Li};
        EnmElement[] actual = mapper
                .readerFor(EnmElement[].class)
                .readValue(mapper.writeValueAsString(expected));
        assertNotNull(actual);
        assertArrayEquals(expected, actual);
    }
}

Немного более сложный пример с использованием перечислений,

import lombok.Data;

@Data
public class CompositeExample {

    private String name;

    private EnmLanguage[] lands;

    private EnmElement[] elements;
}

Пример фрагмента модульного теста,

 @Test
 public void testDeserialize() throws IOException {
     CompositeExample expected = new CompositeExample();

     String name = "Hello";
     expected.setName(name);

     EnmLanguage[] langs = {EnmLanguage.en_US, EnmLanguage.tr_TR};
     expected.setLangs(langs);

     EnmElement[] elements = {EnmElement.H, EnmElement.He,  EnmElement.Li};
     expected.setElements(elements);

     ObjectMapper mapper = new ObjectMapper();
     String json = mapper.writeValueAsString(expected);
     //{"name":"Hello","langs":["en_US","tr_TR"],"elements":["H","He","Li"]}

     CompositeExample actual = mapper
             .readerFor(CompositeExample.class)
             .readValue(json);
     assertNotNull(actual);
     assertEquals(expected, actual);
 }
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...