【问题标题】:how to create a custom JsonDeserializer in Java?如何在 Java 中创建自定义 JsonDeserializer?
【发布时间】:2011-04-10 02:11:37
【问题描述】:

我有一个Map<A,B> fieldOfC 作为 C 类的一个字段。当我尝试使用 Jackson 反序列化 C 时,抛出异常,因为它找不到 Map 的键 A 的反序列化器。所以,我想解决方案是扩展 StdJsonDeserializer 并手动完成。
我的问题是我找不到关于如何使用解析器和我必须实现的“反序列化”方法的上下文的示例。

任何人都可以为这个简单的示例编写代码,以便我可以使用它来构建我真正的反序列化器吗?

public class A{
  private String a1;
  private Integer a2;
}

public class B{
  private String b1;
}

public class C{
  @JsonDeserialize(keyUsing=ADeserializer.class)
  //also tried this: @JsonDeserialize(keyAs=A.class) without success
  private Map<A,B> fieldOfC;
  private String c1;
}

public class ADeserializer extends StdKeyDeserializer {

  protected ADeserializer(Class<A> cls) {
    super(cls);
  }

  protected Object _parse(String key, DeserializationContext ctxt) throws Exception {
    ObjectMapper mapper = new ObjectMapper();
    return mapper.readValue(key, A.class);
  }
}

提前致谢

编辑:谷歌搜索,我发现了一个test,我遇到了同样的问题。这正是我的问题

编辑:当我在方法 findKeyDeserializer(org.codehaus.jackson.map.DeserializationConfig, org.codehaus.jackson.type.JavaType, org.codehaus.jackson.map.BeanProperty) 中读取 here 时,将扩展类从 StdDeserializer 更改为 StdKeyDeserializer

编辑:解决这个问题后,我得到了相关的this one

【问题讨论】:

  • 我知道这还不是一个解决方案,但是 Jackson 1.8 通过注释和模块 API 完全支持自定义密钥序列化器和反序列化器。所以它应该有很大帮助:在早期版本中正确地做这件事是很痛苦的。

标签: java json jackson deserialization


【解决方案1】:

我是 Jackson 的新手,但以下内容对我有用。

首先我给 A 添加一个 JsonCreator 方法:

public class A {
    private String a1;
    private Integer a2;
    public String getA1() { return a1; }
    public Integer getA2() { return a2; }
    public void setA1(String a1) { this.a1 = a1; }
    public void setA2(Integer a2) { this.a2 = a2; }

    @JsonCreator
    public static A fromJSON(String val) throws JsonParseException, JsonMappingException, IOException {
        ObjectMapper mapper = new ObjectMapper();
        A a = mapper.readValue(val,A.class);
        return a;
    }
} 

仅此一项就解决了反序列化问题。对我来说更难的部分是密钥的正确序列化。我在那里所做的是定义一个密钥序列化程序,将命名类序列化为 JSON 序列化,如下所示:

public class KeySerializer extends SerializerBase<Object> {
    private static final SerializerBase<Object> DEFAULT = new StdKeySerializer();

    private Set<Class<?>> objectKeys_ = Collections.synchronizedSet(new HashSet<Class<?>>());

    protected KeySerializer(Class<?>... objectKeys) {
        super(Object.class);
        for(Class<?> cl:objectKeys) {
            objectKeys_.add(cl);
        }
    }


    @Override
    public JsonNode getSchema(SerializerProvider provider, Type typeHint) throws JsonMappingException {
        return DEFAULT.getSchema(provider, typeHint);
    }


    @Override
    public void serialize(Object value, JsonGenerator jgen,
            SerializerProvider provider) throws IOException,
            JsonGenerationException {
        if (objectKeys_.contains(value.getClass())) {
            ObjectMapper mapper = new ObjectMapper();
            StringWriter writer = new StringWriter();
            mapper.writeValue(writer, value);
            jgen.writeFieldName(writer.toString());
        } else {
            DEFAULT.serialize(value, jgen, provider);
        }
    }
}

然后为了证明它有效,序列化和反序列化 C 类的实例:

    ObjectMapper mapper = new ObjectMapper();
    StdSerializerProvider provider = new StdSerializerProvider();
    provider.setKeySerializer(new KeySerializer(A.class));
    mapper.setSerializerProvider(provider);

    StringWriter out = new StringWriter();
    mapper.writeValue(out, c);
    String json = out.toString();
    System.out.println("JSON= "+json);

    C c2 = mapper.readValue(json, C.class);
    System.out.print("C2= ");
    StringWriter outC2 = new StringWriter();
    mapper.writeValue(outC2, c2);
    System.out.println(outC2.toString());

对我来说,这产生了输出:

JSON= {"c1":"goo","map":{"{\"a1\":\"1ccf\",\"a2\":7376}":{"b1":"5ox"},"{\"a1\":\"1cd2\",\"a2\":7379}":{"b1":"5p0"},"{\"a1\":\"1cd5\",\"a2\":7382}":{"b1":"5p3"},"{\"a1\":\"1cd8\",\"a2\":7385}":{"b1":"5p6"}}}
C2= {"c1":"goo","map":{"{\"a1\":\"1ccf\",\"a2\":7376}":{"b1":"5ox"},"{\"a1\":\"1cd2\",\"a2\":7379}":{"b1":"5p0"},"{\"a1\":\"1cd5\",\"a2\":7382}":{"b1":"5p3"},"{\"a1\":\"1cd8\",\"a2\":7385}":{"b1":"5p6"}}}

我觉得应该有更好的方法来说明如何使用注释来序列化密钥,但我无法解决。

【讨论】:

  • 我将尝试将此解决方案与我正在尝试使用 StdKeyDeserializer 做的事情合并,看看是否找到更好的方法。
  • 我使用了你的解决方案,唯一不同的是我在没有使用 objectKeys 属性的情况下实现了序列化方法。我的实现正是你在 if 块中的四行。现在我在 Period joda-time 的课程上遇到了问题,但我会在那个课程上发表一篇新文章。谢谢,西蒙。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-10-19
  • 2011-10-19
  • 2018-04-14
  • 2010-09-05
相关资源
最近更新 更多