当我使用Jackson对该对象进行反序列化时,我试图在Java对象中包含原始JSON。为了测试此功能,我编写了以下测试:
public static class Pojo { public String foo; @JsonRawValue public String bar; } @Test public void test() throws JsonGenerationException, JsonMappingException, IOException { String foo = "one"; String bar = "{\"A\":false}"; Pojo pojo = new Pojo(); pojo.foo = foo; pojo.bar = bar; String json = "{\"foo\":\"" + foo + "\",\"bar\":" + bar + "}"; ObjectMapper objectMapper = new ObjectMapper(); String output = objectMapper.writeValueAsString(pojo); System.out.println(output); assertEquals(json, output); Pojo deserialized = objectMapper.readValue(output, Pojo.class); assertEquals(foo, deserialized.foo); assertEquals(bar, deserialized.bar); }
代码输出以下行:
{"foo":"one","bar":{"A":false}}
JSON正是我想要的外观。不幸的是,当尝试将JSON读回对象时,代码将失败并出现异常。这是例外:
org.codehaus.jackson.map.JsonMappingException:无法从[源:java.io.StringReader@d70d7a;]中的START_OBJECT令牌反序列化java.lang.String实例。行:1,列:13](通过参考链:com.tnal.prism.cobalt.gather.testing.Pojo [“ bar”])
为什么杰克逊在一个方向上只能正常运行,而在另一个方向上却无法运行?似乎它应该能够再次将自己的输出作为输入。我知道我想做的是非正统的(一般建议是为该内部对象创建一个bar名为的属性A),但是我根本不想与此JSON进行交互。我的代码充当该代码的传递对象- 我想接受此JSON并再次发送回去而不用碰任何东西,因为当JSON更改时,我不希望我的代码需要修改。
bar
A
谢谢你的建议。
编辑:使Pojo成为静态类,这将导致另一个错误。
@JsonRawValue仅用于序列化方面,因为反向处理起来有些棘手。实际上,添加它是为了允许注入预编码的内容。
我想可能会增加对反向的支持,尽管那将是很尴尬的:必须先解析内容,然后将其重新写回“原始”形式,该形式可能相同也可能不同(因为字符引用可能会有所不同)。这是一般情况。但这也许对某些问题子集有意义。
但我认为针对您的特定情况的解决方法是将类型指定为“ java.lang.Object”,因为这应该可以正常工作:对于序列化,将按原样输出String,对于反序列化,将其反序列化为一张地图。实际上,如果这样,您可能想要有一个单独的getter / setter。getter将返回String进行序列化(并且需要@JsonRawValue);而setter将采用Map或Object。如果可以的话,您可以将其重新编码为字符串。