如何使用Jackson将原始JSON包含在对象中?


102

当使用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更改时,我不希望我的代码需要修改。

谢谢你的建议。

编辑:使Pojo成为静态类,这将导致另一个错误。

Answers:


64

@JsonRawValue仅用于序列化端,因为反向处理起来有些棘手。实际上,添加它是为了允许注入预编码的内容。

我想可能会增加对反向的支持,尽管那会很尴尬:必须先解析内容,然后将其重新写回“原始”格式,该格式可能相同也可能不同(因为字符引用可能会有所不同)。这是一般情况。但这也许对于某些问题子集是有意义的。

但是我认为针对您的特定情况的解决方法是将类型指定为“ java.lang.Object”,因为这应该可以正常工作:对于序列化,将按原样输出String,对于反序列化,将其反序列化为一张地图。实际上,如果这样,您可能想要有一个单独的getter / setter。getter将返回String进行序列化(并且需要@JsonRawValue);而setter将采用Map或Object。如果可以的话,您可以将其重新编码为字符串。


1
这就像一个魅力。请参阅我对代码的答复(注释中的格式为awgul)。
Yves Amsellem

我为此有一个不同的用例。好像我们不想在deser / ser中生成大量字符串垃圾一样,我们应该能够像这样直接传递一个字符串。我看到了一个跟踪此消息的线程,但似乎没有本机支持。看看markmail.org/message/…–
Sid

@Sid无法同时进行AND标记化。为了支持未处理令牌的传递,将需要额外的状态保持,这使得“常规”解析的效率有所降低。这有点像常规代码和异常抛出之间的优化:支持后者会增加前者的开销。杰克逊的目的不是要使未处理的输入保持可用。最好有它(也用于错误消息),但是需要不同的方法。
StaxMan '16

55

@StaxMan回答之后,我做了以下工作,像个魅力:

public class Pojo {
  Object json;

  @JsonRawValue
  public String getJson() {
    // default raw value: null or "[]"
    return json == null ? null : json.toString();
  }

  public void setJson(JsonNode node) {
    this.json = node;
  }
}

并且,为了忠实于最初的问题,以下是工作测试:

public class PojoTest {
  ObjectMapper mapper = new ObjectMapper();

  @Test
  public void test() throws IOException {
    Pojo pojo = new Pojo("{\"foo\":18}");

    String output = mapper.writeValueAsString(pojo);
    assertThat(output).isEqualTo("{\"json\":{\"foo\":18}}");

    Pojo deserialized = mapper.readValue(output, Pojo.class);
    assertThat(deserialized.json.toString()).isEqualTo("{\"foo\":18}");
    // deserialized.json == {"foo":18}
  }
}

1
我没有尝试,但是它应该起作用:1)使一个JsonNode节点代替Object json 2)使用node.asText()代替toString()。我不确定第二个。
Vadim Kirilchuk

我不知道为什么getJson()还返回一个String。如果它只是返回JsonNode通过setter设置的,它将按需要序列化,不是吗?
对不起,missmissjackson,2016年

@VadimKirilchuk node.asText()返回相反的空值toString()
v.ladynev

36

我可以用自定义解串器要做到这一点(剪切和粘贴来自这里

package etc;

import java.io.IOException;

import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.TreeNode;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;

/**
 * Keeps json value as json, does not try to deserialize it
 * @author roytruelove
 *
 */
public class KeepAsJsonDeserialzier extends JsonDeserializer<String> {

    @Override
    public String deserialize(JsonParser jp, DeserializationContext ctxt)
            throws IOException, JsonProcessingException {

        TreeNode tree = jp.getCodec().readTree(jp);
        return tree.toString();
    }
}

6
太简单了。海事组织,这应该是官方的答案。我尝试了一个非常复杂的结构,其中包含数组,子对象等。也许您可以编辑答案并添加要反序列化的String成员应使用@JsonDeserialize(使用= KeepAsJsonDeserialzier.class)进行注释。(并更正您的班级名称中的拼写错误;-)
Heri 2015年

这适用于反序列化。将原始json序列化为pojo怎么样?如何实现
xtrakBandit 2015年

4
@xtrakBandit用于序列化,使用@JsonRawValue
smartwjw

这就像一个魅力。谢谢Roy和@Heri ..com结合本文以及Heri的评论是恕我直言的最佳答案。
Michal

简单整洁的解决方案。我同意@Heri
mahesh nanayakkara

18

@JsonSetter可能会有所帮助。请参阅我的示例(“数据”应该包含未解析的JSON):

class Purchase
{
    String data;

    @JsonProperty("signature")
    String signature;

    @JsonSetter("data")
    void setData(JsonNode data)
    {
        this.data = data.toString();
    }
}

3
根据JsonNode.toString()文档,该方法将产生开发人员可读的节点表示形式。<b>或可能</ b>不是有效的JSON。因此,这实际上是非常冒险的。
Piotr

@ Piotr,javadoc现在说:“将使用databind的默认设置(作为字符串)生成(从Jackson 2.10起)有效的JSON的方法”
bernie

4

除了为Roy Truelove提供了一个很好的答案之外,这是如何根据以下情况注入自定义解串器的方法@JsonRawValue

import com.fasterxml.jackson.databind.Module;

@Component
public class ModuleImpl extends Module {

    @Override
    public void setupModule(SetupContext context) {
        context.addBeanDeserializerModifier(new BeanDeserializerModifierImpl());
    }
}

import java.util.Iterator;

import com.fasterxml.jackson.annotation.JsonRawValue;
import com.fasterxml.jackson.databind.BeanDescription;
import com.fasterxml.jackson.databind.DeserializationConfig;
import com.fasterxml.jackson.databind.deser.BeanDeserializerBuilder;
import com.fasterxml.jackson.databind.deser.BeanDeserializerModifier;
import com.fasterxml.jackson.databind.deser.SettableBeanProperty;

public class BeanDeserializerModifierImpl extends BeanDeserializerModifier {
    @Override
    public BeanDeserializerBuilder updateBuilder(DeserializationConfig config, BeanDescription beanDesc, BeanDeserializerBuilder builder) {
        Iterator<SettableBeanProperty> it = builder.getProperties();
        while (it.hasNext()) {
            SettableBeanProperty p = it.next();
            if (p.getAnnotation(JsonRawValue.class) != null) {
                builder.addOrReplaceProperty(p.withValueDeserializer(KeepAsJsonDeserialzier.INSTANCE), true);
            }
        }
        return builder;
    }
}

这在Jackson 2.9中不起作用。看起来好像已损坏,因为它现在使用的是PropertyBasedCreator.construct中的旧属性,而不是替换其中一个
dant3

3

这是您的内部类的问题。该Pojo班是一个非静态内部类的测试类的,和杰克逊不能实例化类。因此它可以序列化,但不能反序列化。

像这样重新定义您的班级:

public static class Pojo {
    public String foo;

    @JsonRawValue
    public String bar;
}

注意添加 static


谢谢。这使我更进一步,但是现在我遇到了另一个错误。我将使用新错误更新原始帖子。
bhilstrom 2011年

3

这个简单的解决方案为我工作:

public class MyObject {
    private Object rawJsonValue;

    public Object getRawJsonValue() {
        return rawJsonValue;
    }

    public void setRawJsonValue(Object rawJsonValue) {
        this.rawJsonValue = rawJsonValue;
    }
}

因此,我能够将JSON的原始值存储在rawJsonValue变量中,然后将其(作为对象)与其他字段反序列化为JSON并通过我的REST发送是没有问题的。使用@JsonRawValue并没有帮助我,因为存储的JSON被反序列化为String而不是对象,而这不是我想要的。


3

这甚至可以在JPA实体中使用:

private String json;

@JsonRawValue
public String getJson() {
    return json;
}

public void setJson(final String json) {
    this.json = json;
}

@JsonProperty(value = "json")
public void setJsonRaw(JsonNode jsonNode) {
    // this leads to non-standard json, see discussion: 
    // setJson(jsonNode.toString());

    StringWriter stringWriter = new StringWriter();
    ObjectMapper objectMapper = new ObjectMapper();
    JsonGenerator generator = 
      new JsonFactory(objectMapper).createGenerator(stringWriter);
    generator.writeTree(n);
    setJson(stringWriter.toString());
}

理想情况下,ObjectMapper甚至JsonFactory都来自上下文,并且配置为正确处理JSON(例如,标准或非标准值,例如“ Infinity”浮点数)。


1
根据JsonNode.toString()文档Method that will produce developer-readable representation of the node; which may <b>or may not</b> be as valid JSON.,这实际上是非常冒险的。
Piotr

嗨@Piotr,谢谢您的提示。您当然是对的,这在JsonNode.asText()内部使用并且将输出Infinity和其他非标准JSON值。
乔治

@ Piotr,javadoc现在说:“将使用databind的默认设置(作为字符串)生成(从Jackson 2.10起)有效的JSON的方法”
bernie

2

这是一个完整的工作示例,说明如何使用Jackson模块同时进行@JsonRawValue两种工作(序列化和反序列化):

public class JsonRawValueDeserializerModule extends SimpleModule {

    public JsonRawValueDeserializerModule() {
        setDeserializerModifier(new JsonRawValueDeserializerModifier());
    }

    private static class JsonRawValueDeserializerModifier extends BeanDeserializerModifier {
        @Override
        public BeanDeserializerBuilder updateBuilder(DeserializationConfig config, BeanDescription beanDesc, BeanDeserializerBuilder builder) {
            builder.getProperties().forEachRemaining(property -> {
                if (property.getAnnotation(JsonRawValue.class) != null) {
                    builder.addOrReplaceProperty(property.withValueDeserializer(JsonRawValueDeserializer.INSTANCE), true);
                }
            });
            return builder;
        }
    }

    private static class JsonRawValueDeserializer extends JsonDeserializer<String> {
        private static final JsonDeserializer<String> INSTANCE = new JsonRawValueDeserializer();

        @Override
        public String deserialize(JsonParser p, DeserializationContext ctxt) throws IOException, JsonProcessingException {
            return p.readValueAsTree().toString();
        }
    }
}

然后,您可以在创建以下模块后注册该模块ObjectMapper

ObjectMapper objectMapper = new ObjectMapper();
objectMapper.registerModule(new JsonRawValueDeserializerModule());

String json = "{\"foo\":\"one\",\"bar\":{\"A\":false}}";
Pojo deserialized = objectMapper.readValue(json, Pojo.class);

除了上述之外,您还有其他需要做的事情吗?我发现,ObjectMapper从未调用过JsonRawValueDeserializer的反序列化方法
Michael Coxon,

@MichaelCoxon您设法使其起作用了吗?过去导致我出现问题的一件事是使用org.codehaus.jackson程序包中的注释而没有意识到。确保您所有的进口商品都来自com.fasterxml.jackson
Helder Pereira


1

使用对象在两种方法上都可以正常工作...此方法有两次开销,需要两次将原始值反序列化。

ObjectMapper mapper = new ObjectMapper();
RawJsonValue value = new RawJsonValue();
value.setRawValue(new RawHello(){{this.data = "universe...";}});
String json = mapper.writeValueAsString(value);
System.out.println(json);
RawJsonValue result = mapper.readValue(json, RawJsonValue.class);
json = mapper.writeValueAsString(result.getRawValue());
System.out.println(json);
RawHello hello = mapper.readValue(json, RawHello.class);
System.out.println(hello.data);

RawHello.java

public class RawHello {

    public String data;
}

RawJsonValue.java

public class RawJsonValue {

    private Object rawValue;

    public Object getRawValue() {
        return rawValue;
    }

    public void setRawValue(Object value) {
        this.rawValue = value;
    }
}

1

我遇到了类似的问题,但使用的列表中包含很多JSON itens(List<String>)。

public class Errors {
    private Integer status;
    private List<String> jsons;
}

我使用@JsonRawValue批注管理序列化。但是对于反序列化,我不得不根据Roy的建议创建一个自定义反序列化器

public class Errors {

    private Integer status;

    @JsonRawValue
    @JsonDeserialize(using = JsonListPassThroughDeserialzier.class)
    private List<String> jsons;

}

在下面,您可以看到我的“列表”解串器。

public class JsonListPassThroughDeserializer extends JsonDeserializer<List<String>> {

    @Override
    public List<String> deserialize(JsonParser jp, DeserializationContext cxt) throws IOException, JsonProcessingException {
        if (jp.getCurrentToken() == JsonToken.START_ARRAY) {
            final List<String> list = new ArrayList<>();
            while (jp.nextToken() != JsonToken.END_ARRAY) {
                list.add(jp.getCodec().readTree(jp).toString());
            }
            return list;
        }
        throw cxt.instantiationException(List.class, "Expected Json list");
    }
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.