我正在尝试@JsonValue
和@JsonSerialize
的组合。让我们从我当前的容器类开始:
public class Container {
private final Map<SomeKey, Object> data;
@JsonValue
@JsonSerialize(keyUsing = SomeKeySerializer.class)
public Map<SomeKey, Object> data() {
return data;
}
}
在这种情况下,不使用自定义序列化器
SomeKeySerializer
。如果按以下方式更改容器,则调用序列化器:
public class Container {
@JsonSerialize(keyUsing = SomeKeySerializer.class)
private final Map<SomeKey, Object> data;
}
但是,这不是我想要的,因为这在输出JSON中引入了另一个“数据”级别。
是否可以通过某种方式组合
@JsonValue
和@JsonSerialize
?我总是可以为
Container
编写另一个自定义序列化程序,它或多或少与@JsonValue
背后的功能相同。我认为,这或多或少是一个hack。jackson 版本:2.6.2
最佳答案
这种组合似乎可以满足您的要求:创建一个Converter以从Container中提取Map,然后将@JsonValue添加到SomeKey本身以对其进行序列化:
@JsonSerialize(converter = ContainerToMap.class)
public class ContainerWithFieldData {
private final Map<SomeKey, Object> data;
public ContainerWithFieldData(Map<SomeKey, Object> data) {
this.data = data;
}
}
public static final class SomeKey {
public final String key;
public SomeKey(String key) {
this.key = key;
}
@JsonValue
public String toJsonValue() {
return "key:" + key;
}
@Override
public String toString() {
return "SomeKey:" + key;
}
}
public static final class ContainerToMap extends StdConverter<ContainerWithFieldData, Map<SomeKey, Object>> {
@Override
public Map<SomeKey, Object> convert(ContainerWithFieldData value) {
return value.data;
}
}
@Test
public void serialize_container_with_custom_keys_in_field_map() throws Exception {
ObjectMapper mapper = new ObjectMapper();
assertThat(
mapper.writeValueAsString(new ContainerWithFieldData(ImmutableMap.of(new SomeKey("key1"), "value1"))),
equivalentTo("{ 'key:key1' : 'value1' }"));
}
我根本无法轻松地将容器的访问器方法注释为DTRT,而不是与@JsonValue结合使用。鉴于容器上的@JsonValue基本上是在指定转换器(通过调用带注释的方法实现),这实际上是您所追求的,尽管看起来并不愉快。 (与 jackson 2.6.2一起尝试)
(我从中学到的东西:密钥序列化器与普通序列化器不同,即使它们实现JsonSerializer的方式相同。例如,它们需要在JsonGenerator而不是writeString上调用writeFieldName。在反序列化方面,JsonDeserializer和KeyDeserializer是详细说明的,但不是在序列化方面的,您可以使用@JsonValue从SomeKey制作密钥序列化程序,但不能使用@JsonSerialize(using = ...)注释SomeKey,这让我感到惊讶。