我在资源类中有一个补丁端点,其中一个抽象类作为请求体。我得到了以下错误:
22:59:30 SEVERE [or.ec.ya.in.Unmarshaller] (on Line: 64) (executor-thread-63) Can't create instance这似乎是因为我作为一个论点所声明的身体模型是抽象的,所以它不能反序列化。
我希望能得到Element_A或者Element_B
我如何声明身体是多态的?
这是元素层次结构
public abstract class BaseElement {
public String name;
public Timestamp start;
public Timestamp end;
}
public class Element_A extends BaseElement{
public String A_data;
}
public class Element_B extends BaseElement{
public long B_data;
}这是带有我的端点的资源类。
@Path("/myEndpoint")
public class ResourceClass {
@PATCH
@Path("/{id}")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.TEXT_PLAIN)
public Response updateEvent(@Valid BaseElement element, @Context UriInfo uriInfo, @PathParam long id) {
if (element instanceof Element_A) {
// Element_A logic
} else if (element instanceof Element_B) {
// Element_B logic
}
return Response.status(Response.Status.OK).entity("working").type(MediaType.TEXT_PLAIN).build();
}
}这是我在补丁请求中发送的请求体
{
"name": "test",
"startTime": "2020-02-05T17:50:55",
"endTime": "2020-02-05T17:51:55",
"A_data": "it's my data"
}我还尝试用自定义反序列化器添加@JsonbTypeDeserializer,但没有工作。
@JsonbTypeDeserializer(CustomDeserialize.class)
public abstract class BaseElement {
public String type;
public String name;
public Timestamp start;
public Timestamp end;
}public class CustomDeserialize implements JsonbDeserializer<BaseElement> {
@Override
public BaseElement deserialize(JsonParser parser, DeserializationContext context, Type rtType) {
JsonObject jsonObj = parser.getObject();
String type = jsonObj.getString("type");
switch (type) {
case "A":
return context.deserialize(Element_A.class, parser);
case "B":
return context.deserialize(Element_B.class, parser);
}
return null;
}
}这是我发出的新请求:
{
"type": "A"
"name": "test",
"startTime": "2020-02-05T17:50:55",
"endTime": "2020-02-05T17:51:55",
"A_data": "it's my data"
}抛出此错误:
02:33:10 SEVERE [or.ec.ya.in.Unmarshaller] (executor-thread-67) null
02:33:10 SEVERE [or.ec.ya.in.Unmarshaller] (executor-thread-67) Internal error: null我的pom.xml包括:
<dependency>
<groupId>io.quarkus</groupId>
<artifactId>quarkus-resteasy</artifactId>
</dependency>
<dependency>
<groupId>io.quarkus</groupId>
<artifactId>quarkus-resteasy-jsonb</artifactId>
</dependency>发布于 2020-06-16 03:57:49
JSON目前还没有对多态反序列化的理想支持,因此目前您最多可以做一种解决方案。我们这里有一个设计问题:https://github.com/eclipse-ee4j/jsonb-api/issues/147,所以请投票给它一个+1!
使用自定义反序列化器是正确的,但问题是不能解析当前的JsonObject,然后使用同一个解析器调用context.deserialize(),因为解析器已经超过了读取JSON对象的状态。( JSONParser是一个仅用于前向的解析器,因此没有办法“倒带”它)
因此,您仍然可以在自定义反序列化器中调用parser.getObject(),但是一旦确定了该字符串的具体类型,则需要使用单独的Jsonb实例来解析该特定JSON字符串。
public static class CustomDeserialize implements JsonbDeserializer<BaseElement> {
private static final Jsonb jsonb = JsonbBuilder.create();
@Override
public BaseElement deserialize(JsonParser parser, DeserializationContext context, Type rtType) {
JsonObject jsonObj = parser.getObject();
String jsonString = jsonObj.toString();
String type = jsonObj.getString("type");
switch (type) {
case "A":
return jsonb.fromJson(jsonString, Element_A.class);
case "B":
return jsonb.fromJson(jsonString, Element_B.class);
default:
throw new JsonbException("Unknown type: " + type);
}
}
}备注:您需要将基本对象模型更改为以下内容:
@JsonbTypeDeserializer(CustomDeserialize.class)
public abstract static class BaseElement {
public String type;
public String name;
@JsonbProperty("startTime")
public LocalDateTime start;
@JsonbProperty("endTime")
public LocalDateTime end;
}start和end)以匹配JSON字段名(startTime和endTime),也可以使用@JsonbProperty注释重新映射它们(我在上面已经做过)java.time.LocalDateTime而不是java.sql.Timestamp,因为LocalDateTime处理时区,但是如果您提供的示例数据H 220G 221G 221,两者最终都能工作。https://stackoverflow.com/questions/62398858
复制相似问题