java - 杰克逊反序列化字符串失败
问题描述
我们的系统利用 Jackson (Java 11) 从外部源反序列化对象。当我们收到有效载荷时,它是字符串(UTF-8)格式。它无法反序列化并出现异常 (MismatchedInputException)
无法构造“对象”的实例(尽管至少存在一个 Creator):没有字符串参数构造函数/工厂方法可以从字符串值反序列化(“{
堆栈技术:
- 爪哇 11
- 杰克逊 2.9.8
- 谷歌云(平台)
我们最终使用了多个 replaceAll 语句,只是为了使有效负载处于允许将字符串转换为 jackson 中的对象的状态。代码真的很臭...
我不确定这是否有帮助,但有效负载是从 Google PubSub Subscription 中提取的。所以我有一个消息接收器监听订阅。我最初使用的是 Springs JacksonPubSubMessageConverter,但在上面抛出了异常。我推出了自己的,在下面添加了 replaceAll(..),现在这似乎已经解决了问题。
public class ABCMessageReceiver implements MessageReceiver {
private PubsubMessageConverter converter;
public void receiveMessage(PubsubMessage message, AckReplyConsumer consumer) {
Person person = converter.fromPubSubMessage(message, Person.class);
...
}
}
payload = "{\n \"general\": {\n \"aggregatorId\": \"111111111111\",\n \"communityId\": \"30303030\",\n \"dateStamp\": \"2019-07-09\",\n \"comments\": \"Testing E2E1\"\n },\n \"prospect\": {\n \"firstName\": \"Joe\",\n \"lastName\": \"Smith\",\n \"nickName\": \"\",\n \"email\": \"joe.smith@sample.com\",\n \"gender\": \"MALE\",\n \"maritalStatus\": \"SINGLE\",\n \"dateOfBirth\": \"1956-06-15\",\n \"veteranStatus\": \"NOTAVETERAN\",\n \"address\": {\n \"address1\": \"100 Acme Street \",\n \"address2\": \"Suite 2300\",\n \"city\": \"Acme City\",\n \"state\": \"WI\",\n \"zip\": \"53214\"\n },\n \"phones\": [\n {\n \"number\": \"4145551212\",\n \"type\": \"WORK\"\n }\n ],\n \"financial\": {\n \"budgetAmount\": 2639,\n \"budgetFrequency\": \"MONTHLY\",\n \"medicaid\": true,\n \"medicare\": true,\n \"ltcPolicy\": false,\n \"homeowner\": true,\n \"vaAid\": false\n },\n \"prospectNeeds\": {\n \"desiredCareLevel\": \"AL\"\n }\n },\n \"tour\": {\n \"date\": \"2019-07-14T17:00:00.000Z\",\n \"notes\": \"Testing tour notes 1\"\n }\n}"
payload = payload.replaceAll("\\\\\"", "\"");
payload = payload.replaceAll("\\\\n", "");
payload = payload.substring(1);
payload = payload.substring(0, payload.length()-1);
我不喜欢开销字符串替换,但它现在似乎工作,直到我得到另一个我需要支持的字符串排列。
解决方案
我要感谢@Andreas 和其他提供评论的人。起初它没有连接关于编码,我不得不在第二天回来。是的,如果制作人提交格式正确的 JSON,我会很高兴的。这是我为使其正常工作所做的工作,不是最好的解决方案,但它确实有效。我扩展了 JacksonPubSubMessageConverter.java 并创建了一个新的未编码版本。
UnencodedJacksonPubSubMessageConverter.java
...
public <T> T fromPubSubMessage(PubsubMessage message, Class<T> payloadType) {
try {
String payload = message.getData().toStringUtf8();
payload = removeQuotesIfNecessary(unescapePayload(payload));
return (T) this.objectMapper.readerFor(payloadType).readValue(payload);
}
catch (IOException ex) {
throw new PubSubMessageConversionException("JSON deserialization of an object of type " + payloadType.getName() + " failed.", ex);
}
}
private String removeQuotesIfNecessary(String payload) {
if (payload != null && payload.startsWith("\"") && payload.endsWith("\"")) {
return payload.substring(1).substring(0, payload.length()-2);
}
return payload;
}
private String unescapePayload(String payload) {
if (payload != null)
return StringEscapeUtils.unescapeJson(payload);
return payload;
}