我试图在使用Jackson对该对象进行反序列化时将原始JSON包含在Java对象中。为了测试此功能,我编写了以下测试:
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更改时,我不希望我的代码需要修改。
谢谢你的建议。
编辑:使Pojo成为静态类,这将导致另一个错误。
@JsonRawValue仅用于序列化端,因为反向处理起来有些棘手。实际上,添加它是为了允许注入预编码的内容。
我想可能会增加对反向的支持,尽管那会很尴尬:必须先解析内容,然后将其重新写回“原始”格式,该格式可能相同也可能不同(因为字符引用可能会有所不同)。这是一般情况。但这也许对于某些问题子集是有意义的。
但是我认为针对你的特定情况的解决方法是将类型指定为“ java.lang.Object”,因为这应该可以正常工作:对于序列化,将按原样输出String,对于反序列化,将其反序列化为一张地图。实际上,如果这样,你可能希望有一个单独的getter / setter。getter将返回String进行序列化(并且需要@JsonRawValue);而setter将采用Map或Object。如果可以的话,你可以将其重新编码为字符串。