我需要一个自定义反序列化器来在复杂的POJO中转换字符串。反序列化一直有效,直到使用反序列化器:特别是当使用自定义反序列化器时,我的对象的非对象属性不会被序列化。
我有一个具有pojo作为参数的restful web服务。
public PreventivoResponse calculate(@FormParam(value = "preventivo") PreventivoWs preventivo) throws Exception;
所以我的班级PreventivoWs需要一个fromString(String)
方法。这里是类定义:
@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
@JsonIgnoreProperties(ignoreUnknown = true)
@JsonInclude(JsonInclude.Include.NON_NULL)
public class PreventivoWs implements Serializable{
private static final long serialVersionUID = -554141724349909424L;
private ClienteMultiSelect cliente;
private String riferimento;
private List<EmailWS> email;
private String dataritiro;
private String dataconsegna;
private Long tipoconsegna;
private IndirizzoWS partenza;
private IndirizzoWS destinazione;
List<ColloWs> colli;
HashMap<Long, String> services;
...
}
在jsonObject中,我有一个枚举定义为
{
"value" : "A",
"text" : "Active"
}
但是此对象需要转换反序列化器:
public class TipoPersonaFGJsonDeserializer extends JsonDeserializer<TipoPersonaFG> {
@Override
public TipoPersonaFG deserialize(JsonParser jsonParser, DeserializationContext context)
throws IOException, JsonProcessingException {
JsonToken currentToken = null;
while ((currentToken = jsonParser.nextValue()) != null) {
switch (currentToken) {
case VALUE_STRING:
switch (jsonParser.getCurrentName()) {
case "value":
String name = jsonParser.getText();
return TipoPersonaFG.valueOf(name);
}
break;
default:
break;
}
}
return null;
}
}
并在属性上注释:
@JsonDeserialize(using = TipoPersonaFGJsonDeserializer.class)
private TipoPersonaFG tipo;
fromString方法只需调用jackson ObjectMapper:
public static PreventivoWs fromString(String jsonString) throws IOException{
ObjectMapper mapper = new ObjectMapper();
PreventivoWs oggetto = mapper.readValue(jsonString, PreventivoWs.class);
return oggetto;
}
如果未在jsonString中指定枚举,则它可以正常工作:对象被完美地反序列化; 如果我在jsonString中添加枚举,则反序列化所有对象属性(email,cliente,partenza,destinazione,...),但忽略其他属性(dataritiro,dataconsegna,tipoconsegna)。
为什么呢?自定义解串器打破了反序列化的标准过程?
更新:
自定义反序列化器发生时解析过程被中断:我在json对象的末尾移动了cliente
属性(包含特定的枚举):现在,字段dataconsegna,dataritiro等被反序列化。
因此,当自定义反序列化程序发生时(即使客户对象被中断),反序列化过程也会结束。
答案 0 :(得分:0)
解决! 正如Jackson Wiki所述:
不得处理超出反序列化值的任何令牌(不多或不少)
所以问题出现在反序列化器中:你必须在找到END_OBJECT(})时停止,否则jsonParser继续运行直到流结束,消耗所有其他标记。
@Override
public TipoPersonaFG deserialize(JsonParser jsonParser, DeserializationContext context)
throws IOException, JsonProcessingException {
JsonToken currentToken = null;
String name = null;
while ((currentToken = jsonParser.nextValue()) != null) {
switch (currentToken) {
case VALUE_STRING:
switch (jsonParser.getCurrentName()) {
case "value":
name = jsonParser.getText();
break;
}
break;
case END_OBJECT:
if(name != null)
return TipoPersonaFG.valueOf(name);
else
return null;
}
}
return TipoPersonaFG.valueOf(name);
}
我添加了案例条件 END_OBJECT ,仅使用第一个'}'并正确关闭enum对象的解析。返回在END_OBJECT案例中移动,否则'}'标记将保留在流中,它将关闭枚举的父级。
因此,您需要将对象从“{”标记解析为“}”标记