@JsonView 不会传播到 ("nested") 自定义序列化程序



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

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

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

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

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

但是setConfig((的JavaDoc说"只有当你知道自己在做什么的时候才使用这个方法"——我显然不这么做…

所以我的问题是:

  1. 这种传播JSON视图的方式可以吗
  2. 有更好的方法吗
  3. 有人能解释违约行为吗

(有一些被***标记的行被注释掉了——这些只有在没有自定义序列化程序的情况下才有趣(在这种情况下,JSON视图似乎被传播了(

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();
}
}
}

Active应通过SerializerProvider正确传播。您是对的,您不应该(也不必(以这种方式设置视图——例如,它不是线程安全的,并且ObjectMapper的所有配置都要在使用之前完成(ObjectReaderObjectWriter允许通过各种工厂方法进行每次调用的更改(。

为什么会发生这种情况尚不清楚:我认为用版本信息和完整复制(如果它比代码更多的话(来提交一个bug是有意义的。

最新更新