使用jackson库反序列化MongoDB $ oid

时间:2013-09-27 21:10:59

标签: java json mongodb

我有一个非常简单的POJO,我写信给MongoDB。然后它返回包含_id属性作为$ oid对象的JSON。我想把它写进我的POJO,但到目前为止都失败了。我已经看到很多人都在努力解决这个问题并建议不同的解决方案,但我还没有让他们中的任何一个工作。可能是因为没有人提供他们实际使用的库的确切细节。有codehaus.jackson。*有com.fastxml.jackson。,jongo,org.mongo。,com.mongo。* etc。

我的对象如下:

package com.blueplanetsoftware.metrics.rest;

import javax.persistence.Id;

import com.fasterxml.jackson.annotation.JsonProperty;

public class MongoObject
{
    @JsonProperty("_id")
    private String _id;

    @JsonProperty("userID")
    private String userID;
    @JsonProperty("appID")
    private String appID;  
    @JsonProperty("message")
    private String message = "";
    @JsonProperty("session")
    private String session;

    public MongoObject() {}

    /**
     * @return the id
     */
    @Id
    @JsonProperty("_id")
    public String getId()
    {
        return _id;
    }

    /**
     * @param id the id to set
     */
    @Id
    @JsonProperty("_id")
    public void setId(String id)
    {
        this._id = id;
    }
}

我回来的JSON看起来像这样:

{ 
  "userID" : "aap" ,
  "appID" : "noot" , 
  "message" : "JSON" , 
  "session" : "ea944555b5ea8ea6" , 
  "_id" : { "$oid" : "5245f1063004348555e54815"}
}

当我尝试使用此代码将JSON反序列化到我的类中时:

com.fasterxml.jackson.databind.ObjectMapper mapper = new com.fasterxml.jackson.databind.ObjectMapper();
MongoObject o = mapper.readValue(response, MongoObject.class);

它给了我一个例外:

java.lang.AssertionError: Can not deserialize instance of java.lang.String out of START_OBJECT token
 at [Source: java.io.StringReader@456c1227; line: 1, column: 93] (through reference chain: com.blueplanetsoftware.metrics.rest.MongoObject["_id"])

那么如何将_id / $ oid字符串放入我的_id属性?我很惊讶这不是一个简单的方法。至少到目前为止我还没有找到任何东西。

1 个答案:

答案 0 :(得分:4)

你试过吗?

@ObjectId @Id 
private String id;

我已经写了一个测试,看看它是否有效:

@Test
public void shouldConvertJSONStringIntoPojo() throws IOException {
    ObjectMapper mapper = new ObjectMapper();
    MongoObject actual = mapper.readValue("{\n" +
                                     "        \"userID\" : \"aap\" ,\n" +
                                     "        \"appID\" : \"noot\" ,\n" +
                                     "        \"message\" : \"JSON\" ,\n" +
                                     "        \"session\" : \"ea944555b5ea8ea6\" ,\n" +
                                     "        \"_id\" : { \"$oid\" : \"5245f1063004348555e54815\"}\n" +
                                     "}", MongoObject.class);

    assertThat(actual, equalTo(new MongoObject("5245f1063004348555e54815", "aap", "noot", "JSON", "ea944555b5ea8ea6")));
}

我正在使用的库:

import com.fasterxml.jackson.databind.ObjectMapper; (jackson-databind-2.1.4,jar)
import org.junit.Test;                              (junit-4.11.jar)
import static org.hamcrest.CoreMatchers.equalTo;    (hamcrest-core-1.3.jar)
import static org.junit.Assert.assertThat;          (junit-4.11.jar)

我对问题中发布的MongoObject所做的唯一更改是添加一个带有所有字段的构造函数,hashCode,等于& toString(所以我可以测试它)。如果我用那个MongoObject运行那个测试,那么我会得到和你一样的错误:

com.fasterxml.jackson.databind.JsonMappingException: Can not deserialize instance of java.lang.String out of START_OBJECT token
 at [Source: java.io.StringReader@7a7ff2aa; line: 5, column: 40] (through reference chain: MongoObject["_id"])
    at com.fasterxml.jackson.databind.JsonMappingException.from(JsonMappingException.java:164)
    at com.fasterxml.jackson.databind.DeserializationContext.mappingException(DeserializationContext.java:599)

但是如果我对你的MongoObject进行一次小改动:

public class MongoObject
{
    @JsonProperty("_id")
    @org.jongo.marshall.jackson.oid.ObjectId
    private String _id;

    @JsonProperty("userID")
    private String userID;
    @JsonProperty("appID")
    private String appID;
    @JsonProperty("message")
    private String message;
    @JsonProperty("session")
    private String session;

    public MongoObject() {}

//... insert constructors, hashCode, equals and toString just for testing...
}

然后测试通过。