首页 > 解决方案 > 杰克逊反序列化字符串失败

问题描述

我们的系统利用 Jackson (Java 11) 从外部源反序列化对象。当我们收到有效载荷时,它是字符串(UTF-8)格式。它无法反序列化并出现异常 (MismatchedInputException)

无法构造“对象”的实例(尽管至少存在一个 Creator):没有字符串参数构造函数/工厂方法可以从字符串值反序列化(“{

堆栈技术:

我们最终使用了多个 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);

我不喜欢开销字符串替换,但它现在似乎工作,直到我得到另一个我需要支持的字符串排列。

标签: javajsonspringjackson

解决方案


我要感谢@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;
}

推荐阅读