9

我使用 Jackson 作为工具来声明一些我无法注释(或根本无法修改)其类的对象。其中一个类具有用于无类型列表的 setter 和 getter。这是一个经过消毒的版本:

public class Family {
    private List members;
    public List getMembers() { return members; }
    public void setMembers(List members) { this.members = members; }
    //...many, many other properties
}

public class Member {
    private String name;
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
}

这是我试图反序列化的 JSON:

{ "members" : [ { "name" : "Mark" } ] }

我会使用的天真代码是这样的:

ObjectMapper mapper = new ObjectMapper();
Family family = mapper.readValue(json, Family.class);
Member member = (Member) family.getMembers().get(0);
System.out.println(member.getName());

但是当然这失败了,因为杰克逊不知道创建一个列表Members而不是它的后备,一个LinkedHashMaps 的列表。

指示杰克逊将其members视为的最简单方法是List<Member>什么? 我不认为我想为该类使用完全自定义的反序列化器,因为 Jackson 可以很好地处理许多其他属性。

这是我能想到的最好的,使用 BeanDeserializerModifier:

mapper.setDeserializerProvider(new StdDeserializerProvider()
  .withDeserializerModifier(new BeanDeserializerModifier() {
      @Override
      public BeanDeserializerBuilder updateBuilder(DeserializationConfig config, BasicBeanDescription beanDesc, BeanDeserializerBuilder builder) {
          if (beanDesc.getBeanClass() == Family.class) {
            CollectionType type = CollectionType.construct(ArrayList.class, SimpleType.construct(Member.class));
            TypeDeserializer typeDeserializer = type.getTypeHandler();
            SettableBeanProperty.MethodProperty membersProperty = (SettableBeanProperty.MethodProperty) builder.removeProperty("members");
            builder.addProperty(new SettableBeanProperty.MethodProperty(
               "members",
               type,
               typeDeserializer,
               beanDesc.getClassAnnotations(),
               (AnnotatedMethod) membersProperty.getMember()
            ));
          }
          return builder;
      }}));

它有效,但对于我正在尝试做的事情来说似乎真的很低级(而且很冗长!)。我在这里想念什么?

编辑

我应该注意,我使用的是 Jackson 1.8.2,但如果有令人信服的理由可以更新。

4

1 回答 1

9

混合注释是我遗漏的难题的关键部分。这是解决此问题的更清洁的方法:

ObjectMapper mapper = new ObjectMapper();
mapper.getDeserializationConfig().addMixInAnnotations(Family.class, FamilyMixin.class);

Family family = mapper.readValue(json, Family.class);
Member member = (Member) family.getMembers().get(0);

//...

interface FamilyMixin {
    @JsonDeserialize(contentAs = Member.class)
    void setMembers(List members);
}

混合注释让您可以对受您控制的代理进行注释。当将该混合类应用于真实类时,Jackson 的行为就好像这些注释注释了真实类的成员一样。

在我的例子中,我JsonDeserialize.contentAs()用来指定容器的内容类型。但我相信大多数注释应该可以使用这种方法。

于 2012-05-31T21:09:06.877 回答