8

我有一个包含 Map (带有非 String 键)和其他一些字段的类。

public class MyClass() {
    private Map<KeyObject, OtherObject> map;
    private String someField;

    public MyClass(Map<KeyObject, OtherObject> map, String someField) {
        this.map = map;
        this.someField = someField;
    }

    // Getters & Setters
}

我想使用杰克逊序列化和反序列化这个类。我看到了一种不同的方法,并决定尝试使用杰克逊模块

我关注了这篇文章并扩展了 JsonDeserializer 和 JsonSerializer。问题是应该键入这些类,所以它应该看起来像

public class keyDeserializer extends JsonDeserializer<Map<KeyObject, OtherObject>> {
...
}

KeySerializer 也是如此。

然后添加到模块:

module.addSerializer(new keySerializer());
module.addDeserializer(Map.class, new keyDeserializer());

但这显然是错误的,因为我遇到了一个例外:

keySerializer does not define valid handledType() -- must either register with method that takes type argument  or make serializer extend 'org.codehaus.jackson.map.ser.std.SerializerBase'

我可以将我的序列化器和反序列化器输入到MyClass,但是我不得不手动解析所有这些,这是不合理的。

更新:

我设法通过使用注释绕过代码中的模块创建

@JsonDeserialize(using = keyDeserializer.class)
@JsonSerialize(using = keySerializer.class)
private Map<KeyObject, OtherObject> map;

但是我必须自己从 toString() 输出序列化/反序列化整个映射结构。所以尝试了不同的注释:

@JsonDeserialize(keyUsing = MyKeyDeserializer.class)
private Map<KeyObject, OtherObject> map;

MyKeyDeserializer 扩展org.codehaus.jackson.map.KeyDeserializer并覆盖该方法的位置

public Object deserializeKey(String key, DeserializationContext ctxt) throws IOException, JsonProcessingException {...}

然后手动反序列化我的密钥,但再次从我的密钥类的 toString() 输出。

这不是最优的(这种依赖于 toString() 方法)。有没有更好的办法?

4

1 回答 1

13

最终使用了这个序列化程序:

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

    protected MapKeySerializer() {
    super(Object.class);
    }

    @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 (null == value) {
        throw new JsonGenerationException("Could not serialize object to json, input object to serialize is null");
    }
    StringWriter writer = new StringWriter();
    mapper.writeValue(writer, value);
    jgen.writeFieldName(writer.toString());
    }
}

这个反序列化器:

public class MapKeyDeserializer extends KeyDeserializer {

    private static final ObjectMapper mapper = new ObjectMapper();

    @Override
    public Object deserializeKey(String key, DeserializationContext ctxt) throws IOException, JsonProcessingException {
    return mapper.readValue(key, MyObject.class);
    }
}

注释我的地图:

@JsonDeserialize(keyUsing = MapKeyDeserializer.class)
@JsonSerialize(keyUsing = MapKeySerializer.class)
private Map<KeyObject, OtherObject> map;

这是对我有用的解决方案,希望这对其他人有帮助。

于 2014-08-19T09:33:11.163 回答