如何基于共同领域杰克逊抽象地反序列化不同的枚举类型?

Cap*_*Man 5 java generics enums json jackson

问题:

我在enum杰克逊的反序列化中与代码中的名字不符,下面是json的样本.

{
    "thing1": {"foo": "cool-guy"},
    "thing2": {"foo": "loser-face"}
}
Run Code Online (Sandbox Code Playgroud)

这是enum,我interface稍后会解释.

enum Foo implements HasText {
    COOL_GUY("cool-guy"), LOSER_FACE("loser-face"), // etc...

    private String text;

    private Foo(String text) {
        this.text = text;
    }

    @Override
    public String getText() {
        return text;
    }
}
Run Code Online (Sandbox Code Playgroud)

我知道如何通过在foo的setter方法上创建反序列化器(下面)和注释来为每个enum 单独解决这个问题@JsonDeserialize(using = FooDeserializer .class).

public class FooDeserializer extends JsonDeserializer<Enum<Foo>> {
    @Override
    public Foo deserialize(JsonParser p, DeserializationContext context) throws Exception {

      if (p.getCurrentToken().equals(JsonToken.VALUE_STRING)) {
        String jsonText = p.getText();
        Stream<Foo> stream = Arrays.asList(Foo.values()).stream();
        return stream.filter(a -> a.getText().equals(jsonText.toLowerCase())).findAny().get();
      }

      throw context.mappingException(Foo.class);
    }
}
Run Code Online (Sandbox Code Playgroud)

题:

有没有办法抽象地这样做?这就是为什么我将HasText添加interface到我的所有枚举中,希望有办法做这样的事情:

public class EnumWithTextDeserializer<T extends Enum<T> & HasText> extends JsonDeserializer<T> {

    @Override
    public T deserialize(JsonParser p, DeserializationContext context) throws Exception {
        if (p.getCurrentToken().equals(JsonToken.VALUE_STRING)) {
          final String jsonText = p.getText();
          final Stream<T> stream = Arrays.asList(runtimeClass().getEnumConstants()).stream();
          return stream.filter(a -> a.getText().equals(jsonText.toLowerCase())).findAny().get();
        }
        throw context.mappingException(runtimeClass());
    }

    @SuppressWarnings("unchecked")
    private Class<T> runtimeClass() {
        ParameterizedType superclass = (ParameterizedType) getClass().getGenericSuperclass();
        return (Class<T>) superclass.getActualTypeArguments()[0];
    }
}
Run Code Online (Sandbox Code Playgroud)

编译不会让我@JsonDeserialize(using = EnumWithTextDeserializer.class)用这个类注释setter方法(),因为

Type mismatch: cannot convert from Class<EnumWithTextDeserializer> to Class<? extends JsonDeserializer<?>>".
Run Code Online (Sandbox Code Playgroud)

实际上,我想要做的就是enum基于getText()方法对这些s 进行反序列化.

ale*_*wen 1

为了反序列化,您可以使用指定字符串值@JsonValue

public enum FooEnum implements WithText {
    AWESOME("awesome-rad"),
    NARLY("totally-narly");

    private final String text;

    FooEnum(String text) {
        this.text = text;
    }

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

}
Run Code Online (Sandbox Code Playgroud)

然后执行此代码进行序列化/反序列化

    ImmutableMap<String, FooEnum> map = ImmutableMap.of("value", FooEnum.AWESOME, "value2", FooEnum.NARLY);
    final String value;
    try {
        value = objectMapper.writeValueAsString(map);
    } catch (JsonProcessingException e) {
        throw Throwables.propagate(e);
    }

    Map<String, FooEnum> read;
    try {
        read = objectMapper.readValue(value, new TypeReference<Map<String, FooEnum>>() {});
    } catch (IOException e) {
        throw Throwables.propagate(e);
    }
Run Code Online (Sandbox Code Playgroud)

我得到:

read = {LinkedHashMap@4627}  size = 2
  0 = {LinkedHashMap$Entry@4631} "value1" -> "AWESEOME"
  1 = {LinkedHashMap$Entry@4632} "value2" -> "NARLY"
Run Code Online (Sandbox Code Playgroud)