@JsonView没有传播到(“嵌套”)自定义序列化程序

时间:2018-11-26 16:00:32

标签: java spring jackson json-view

在下面的单元测试中,有两个bean:beanA和beanB,其中beanA包含对beanB的引用。

当使用自定义序列化程序对提供JSON View的bean进行序列化 时,beanB的自定义序列化程序不知道所提供的JSON View,因为测试的输出为:

{
  "clazz" : "BeanA",
  "activeView" : "ViewTest",
  "beanB" : {
    "clazz" : "BeanB",
    "activeView" : null
  }
}

我发现传播activeView的唯一方法是:(请参见下面的完整源代码)

ObjectMapper mapper = (ObjectMapper) gen.getCodec();
mapper.setConfig(provider.getConfig());

但是setConfig()的JavaDoc说“只有在知道自己在做什么的情况下才使用此方法”-我显然不这样做...

所以我的问题是:

  1. 通过这种方式传播JSON View还可以吗?
  2. 有更好的方法吗?
  3. 有人可以解释默认行为吗?

(有一些用***标记的行注释掉了-只有在没有自定义序列化程序的情况下(在这种情况下JSON View似乎会传播),这些才有意义。)

public class ViewTest {

    @Test
    public void hello() throws JsonProcessingException {

        BeanA beanA = new BeanA();
        beanA.beanB = new BeanB();

        String result = new ObjectMapper()
//            .disable(MapperFeature.DEFAULT_VIEW_INCLUSION) // ***
            .writerWithView(ViewTest.class)
            .withDefaultPrettyPrinter()
            .writeValueAsString(beanA);

        System.out.println(result);
    }

    @JsonSerialize(using = BeanASerializer.class)
    static public class BeanA {

//        @JsonView(ViewTest.class) // ***
        public String clazz = this.getClass().getSimpleName();

//        @JsonView(ViewTest.class) // ***
        public BeanB beanB;
    }

    @JsonSerialize(using = BeanBSerializer.class)
    static public class BeanB {

//        @JsonView(ViewTest.class) // ***
        public String clazz = this.getClass().getSimpleName();
    }

    static void writeClazzAndActiveView(String clazz, JsonGenerator gen, SerializerProvider provider) throws IOException {
        gen.writeStringField("clazz", clazz);
        gen.writeFieldName("activeView");
        if (provider.getActiveView() == null) {
            gen.writeNull();
        } else {
            gen.writeString(provider.getActiveView().getSimpleName());
        }
    }

    static class BeanASerializer extends JsonSerializer<BeanA> {

        @Override
        public void serialize(BeanA bean, JsonGenerator gen, SerializerProvider provider) throws IOException, JsonProcessingException {

            gen.writeStartObject();
            writeClazzAndActiveView(bean.clazz, gen, provider);

            // the only way I found to propagate the activeView:
//            ObjectMapper mapper = (ObjectMapper) gen.getCodec();
//            mapper.setConfig(provider.getConfig());

            gen.writeObjectField("beanB", bean.beanB);
            gen.writeEndObject();
        }
    }

    static class BeanBSerializer extends JsonSerializer<BeanB> {

        @Override
        public void serialize(BeanB bean, JsonGenerator gen, SerializerProvider provider) throws IOException, JsonProcessingException {

            gen.writeStartObject();
            writeClazzAndActiveView(bean.clazz, gen, provider);
            gen.writeEndObject();
        }
    }
}

1 个答案:

答案 0 :(得分:1)

应该通过SerializerProvider正确传播Active。没错,您不应该那样设置(也不必设置)视图,因为它不是线程安全的,ObjectMapper的所有配置都必须在任何使用之前完成 ({ObjectReaderObjectWriter允许通过各种工厂方法更改每次通话)。

为什么会发生这种情况尚不清楚:我认为使用版本信息和完整复制(如果代码中包含的内容多于代码)提交错误是有道理的。