我有一个CloudEvent<T>类,它使用Jackson (2.9.0 -最后一个版本)进行多态反序列化,如下所示:
@Data
@NoArgsConstructor
@AllArgsConstructor
public class CloudEvent<T> {
@NonNull
private String eventType;
@JsonTypeInfo(
use = JsonTypeInfo.Id.NAME,
include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
property = "eventType",
defaultImpl = Void.class)
@JsonSubTypes({
@JsonSubTypes.Type(value = MyEvent1.class, name = "event-1"),
@JsonSubTypes.Type(value = MyEvent2.class, name = "event-2")
})
private T data;
}然后反序列化为:
String cloudEventJson1 = "{\"eventType\":\"event-1\",\"data\":{\"id\":\"123\",\"details\":\"detail1\"}}";
CloudEvent deserializedEvent1 = objectMapper.readValue(cloudEventJson1, CloudEvent.class); //without subtypes这一切都很好。但是由于一些限制,我不能在CloudEvent类上使用注释(由外部依赖关系提供)。
因此,我将ObjectMapper配置如下:
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.registerSubtypes(new NamedType(MyEvent1.class, "event-1"));
objectMapper.registerSubtypes(new NamedType(MyEvent2.class, "event-2"));
TypeResolverBuilder typeResolverBuilder = new ObjectMapper.DefaultTypeResolverBuilder(ObjectMapper.DefaultTyping.OBJECT_AND_NON_CONCRETE)
.init(JsonTypeInfo.Id.NAME, null) //CLASS works
.inclusion(JsonTypeInfo.As.EXTERNAL_PROPERTY)
.typeProperty("eventType")
.typeIdVisibility(true)
// .defaultImpl(Void.class);
objectMapper.setDefaultTyping(typeResolverBuilder);但是,使用与上面相同的方法反序列化不起作用。它正在读取eventType,但它无法与已注册的子类型匹配。我不能在反序列化中使用泛型或TypeReferance,因为我需要使用spring-integration读取只接受主类的事件;反序列化之后手动完成模式匹配。
例外:
com.fasterxml.jackson.databind.exc.InvalidTypeIdException: Could not resolve type id 'event-1' as a subtype of [simple type, class java.lang.Object]: known type ids = [] (for POJO property 'data')
at [Source: (String)"{"eventType":"event-1","data":{"id":"123","details":"detail1"}}"; line: 1, column: 271]这也是为所有输入类配置ObjectMapper。是否可以将这个typeResolverBuilder和subtypes连接到CloudEvent.class (就像注释方法那样)。
发布于 2019-03-19 14:33:17
即使不能修改类,仍然可以依赖注释。Jackson支持一个名为混合的特性:您可以认为它是一种面向方面的方式,可以在运行时添加更多的注释,以增强静态定义的注释。
首先,按以下方式定义接口:
public interface CloudEventMixIn<T> {
@JsonTypeInfo(
use = JsonTypeInfo.Id.NAME,
include = JsonTypeInfo.As.EXTERNAL_PROPERTY,
property = "eventType",
defaultImpl = Void.class)
@JsonSubTypes({
@JsonSubTypes.Type(value = MyEvent1.class, name = "event-1"),
@JsonSubTypes.Type(value = MyEvent2.class, name = "event-2")
})
public T getData();
}然后将ObjectMapper配置为使用定义的接口作为实际类/接口的混合:
ObjectMapper mapper = new ObjectMapper();
mapper.addMixIn(CloudEvent.class, CloudEventMixIn.class);来自addMixIn(Class target, Class mixinSource)方法文档:
方法用于添加混合注释,用于增强指定的类或接口。来自
mixinSource的所有注释都被用来覆盖target(或其超级类型)所具有的注释。
https://stackoverflow.com/questions/55243350
复制相似问题