手动参考支持

时间:2013-02-18 15:12:50

标签: java mongodb jongo

考虑课程:

class Person {
    String name;
    Person father;
    Person mother;
    List<Person> children;
}

有没有办法向jongo表明fathermotherchildren应该是manual references到同一集合中的其他对象而不是嵌入对象?

注意:这与DBRefs不同。

2 个答案:

答案 0 :(得分:1)

还没有。

最简单的方法是手动发出第二个查询。

尽管如此,您可以创建一个Jackson de / serializer来在解组过程中获取文档。 几个月前,我们已经创建了一个在无法处理期间获取DBRef的峰值,这段代码可以提供帮助:Handle DBRef during (un)marshalling

随意添加feature request

答案 1 :(得分:1)

我用一种方法来处理它。不确定它是否完美,但它适用于我自己。

假设您有这些课程

public class MyObject {

    @Id
    private String id;
    private MyEmbeddedObject embeddedObject;

    public String getId() {
        return this.id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public MyEmbeddedObject getEmbeddedObject() {
        return this.embeddedObject;
    }

    public void setEmbeddedObject(MyEmbeddedObject embeddedObject) {
        this.embeddedObject = embeddedObject;
    }

}


public class MyEmbeddedObject {

    @Id
    private String id;

    public String getId() {
        return this.id;
    }

    public void setId(String id) {
        this.id = id;
    }

}

您希望将MyEmbeddedObject存储在集合“embed”中,将Object存储在集合“object”中,并手动引用“embed”集合ID。例如:

object : { id:1, embeddedObjectId: 99 }
embed : { id : 99 }

您可以为MyEmbeddedObject

实现自定义序列化程序和反序列化程序
public class MyEmbeddedObjectSerializer extends JsonSerializer<MyEmbeddedObject> {

    @Override
    public void serialize(MyEmbeddedObject value, JsonGenerator jgen, SerializerProvider provider) throws IOException, JsonProcessingException {
        jgen.writeText(value.getId());
    }
}

public class MyEmbeddedObjectDeserializer extends JsonDeserializer<MyEmbeddedObject> {

    @Override
    public MyEmbeddedObject deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException, JsonProcessingException {
        MyEmbeddedObject obj = new MyEmbeddedObject();
        obj.setId(jsonParser.getString());
        return new MyEmbeddedObject();

    }
 }

如果您在Jongo中注册序列化程序,问题是它将作为MyObject的成员用于MyEmbeddedObject,而当它本身存储在“embed”集合中时也用于MyEmbeddedObject。

仅当MyEmbeddedObject嵌入MyOBject时才应用自定义serizalizers /反序列化器,我使用Mixin注释。

public interface MyObjectMixIn {

    @JsonSerialize(using=MyEmbeddedObjectSerializer .class)
    public MyEmbeddedObject getEmbeddedObject();

    @JsonDeserialize(using=MyEmbeddedObjectDeserializer.class)
    public void setEmbeddedObject(MyEmbeddedObject embeddedObject);

}

然后定义一个模块

public class MyModule extends SimpleModule {
    @Override
    public void setupModule(SetupContext context) {
        context.setMixInAnnotations(MyObject.class, MyObjectMixIn.class);
    }
}

最后在Jongo注册模块

Jongo jongo = new Jongo(db, JacksonMapper.Builder().registerModule(new MyModule ()).build());

现在用它来存储Mongo

  1. 将embeddedObject单独存储在集合“embed”中以获取id
  2. 确保object的embeddedObject成员包含id
  3. 将对象存储在集合“object”中。只存储嵌入对象的id
  4. 并阅读数据存储区

    1. 从集合“对象”中读取对象。 embeddedobject将仅使用其ID
    2. 进行初始化
    3. 如果需要,请阅读集合“embed”中的embeddedObject。
    4. 编辑:为了避免延迟加载并在反序列化时加载完整的对象,来自Jackson的HandlerInstantiator对象可用于自定义序列化器和反序列化器构造机制。

      希望有所帮助。