多个相同类型的杰克逊序列化器

时间:2013-11-17 23:40:06

标签: java json jackson

我正在使用Jackson进行JSON序列化,并编写了几个自定义String序列化程序,一个用于类的每个getter方法。每个方法返回相同的类型Set<String>,但每个都使用不同的序列化程序。

不幸的是,杰克逊没有使用每个序列化程序,每个方法一个,但是两个都使用一个序列化程序。它似乎采用按字母顺序排列的任何方法,并使用其序列化程序用于这两种方法。我期望的是第一种方法注释的序列化器用于第一种方法,第二种方法注释的序列化器用于第二种方法。调试似乎表明杰克逊在地图中的序列化器具有方法的返回类型(两者都相同)。

示例:

public class FooBar {

  private Set<String> foos = new HashSet<String>();
  private Set<String> bars = new HashSet<String>();

  @JsonProperty("FooWrapper")
  @JsonSerialize(contentUsing = FooSerializer.class)
  public Set<String> getFoos() {
    return foos;
  }

  @JsonProperty("BarWrapper")
  @JsonSerialize(contentUsing = BarSerializer.class)
  public Set<String> getBars() {
    return bars;
  }
}

有关如何使用getFoos()FooSerializer方法序列化以及使用getBars()序列化的BarSerializer方法的任何建议?在此示例中,为两个方法调用BarSerializer

注意,如果我将其中一个方法的签名更改为另一个集合类型,以便它们不同 - 例如List<String> - 序列化可以正常工作。

提前致谢。

1 个答案:

答案 0 :(得分:7)

我认为在将ObjectMapper@JsonSerialize(contentUsing = BarSerializer.class)结合使用时,您在1.9.xx版本中无法实现的目标。

杰克逊确实缓存了序列化程序,并根据与序列化程序关联的JavaType (在本例中为Set<String>对其进行缓存。请参阅StdSerializerProvider.findValueSerializer(JavaType valueType, BeanProperty property)

虽然BeanProperty传递给此方法,但它不会用作缓存键的一部分。您可以继承StdSerializerProvider并在缓存值序列化程序时考虑BeanProperty参数,但这可能不是解决问题的最简单方法。

快速解决方法是使用@JsonSerialize(using = FooCollectionSerializer.class)并自行处理序列化。通过执行此操作,序列化程序directly coupled到用于序列化属性的BeanPropertyWriter。使用@JsonSerialize(contentUsing = BarSerializer.class)时,没有与BeanPropertyWriter耦合的序列化程序触发serializer lookup根据JavaType

缓存序列化程序
public class FooBar {

    private Set<String> foos = new HashSet<>();
    private Set<String> bars = new HashSet<>();

    @JsonProperty("FooWrapper")
    @JsonSerialize(using = FooCollectionSerializer.class)
    public Set<String> getFoos() {
        return foos;
    }

    @JsonProperty("BarWrapper")
    @JsonSerialize(using = BarCollectionSerializer.class)
    public Set<String> getBars() {
        return bars;
    }

    public static class FooCollectionSerializer extends JsonSerializer<Collection<String>> {

        JsonSerializer<Collection<String>> serializer;

        public FooCollectionSerializer() {
            //let Jackson deal with serializing the collection and just specify how you want to serialize indivial items
            this.serializer = new StringCollectionSerializer(null, new FooSerializer());
        }

        @Override
        public void serialize(Collection<String> value, JsonGenerator jgen, SerializerProvider provider) throws IOException, JsonProcessingException {
            serializer.serialize(value, jgen, provider);
        }
    }

    public static class FooSerializer extends SerializerBase<String> {
        public FooSerializer() {
            super(String.class);
        }

        @Override
        public void serialize(String value, JsonGenerator jgen, SerializerProvider provider) throws IOException {
            jgen.writeString(value);
        }
    }

    public static class BarCollectionSerializer extends JsonSerializer<Collection<String>> {

        @Override
        public void serialize(Collection<String> values, JsonGenerator jgen, SerializerProvider provider) throws IOException, JsonProcessingException {
            //handle serializing the collection yourself
            jgen.writeStartArray();
            for (String value : values) {
                jgen.writeString(value);

            }
            jgen.writeEndArray();
        }
    }
}