Spring Data MongoDB动态字段映射

4

我在我的Java类中有这个模型

@Document
public class Template {

    private String type;

    private String code;

    @Version
    Long version;
}

我需要添加一个名为template的新字段,并将此字段映射为动态,换句话说,我会像这样对文档进行建模

{
  _id: 'id' 
  type:'myType',
  code:'myCode'
  template:{
    someFiled:[
      {
        subField1:'value1',
        subField2:'value2'
      },
      {
        sub1Field1:'1value1',
        sub1Field2:'1value2'
      }
      .......................
    ],
    otherField:[
      {
        otherField1:'value1',
        otherField2:'value2'
      }
    ],
    .........
  },
  version:1000L
}

有没有一种方法可以将字段注释为动态的?

解决方案

    @Document
    public class Template {

        private String type;

        private String code;

        private Map<String, Object> template;

        @Version
        Long version;
    }

当你说“动态”时,你确切指的是什么:“模式事先未知”还是“它具有嵌套结构”或两者混合或其他某些东西? - Marc Tarin
@MarcTarin 我事先不知道模式,但基本上它是一个嵌套结构。 - Claudio Pomo
添加一个JsonObject作为字段怎么样?这样您就可以将任何文档添加到其中了。 - user1211
1
未定义的模式已经在这里处理:这里这里这里...你的嵌套结构看起来像是一个Map<String, List<Map<String, String>>>,但我会使用BasicDBObject - Marc Tarin
1个回答

6

我找到了一个完美的解决方案。以我的项目为例:

@Data
@Document(collection = "logs")
public class Log {
    @Id
    private String id;
    private Object data;

    // data field can be a string
    public void setData(String str) {
        data = str;
    }
    // data field can be a {}
    public void setData(JsonObject jsonObject) {
        data = new BasicDBObject(jsonObject.getMap());
    }
    // data can be a []
    public void setData(JsonArray jsonArray) {
        BasicDBList list = new BasicDBList();
        list.addAll(jsonArray.getList());
        data = list;
    }
}

data字段声明为Object类型,并为其实现三种setter。

以下是测试用例:

@RunWith(SpringRunner.class)
@SpringBootTest
public class LogRepositoryTest {

    @Autowired
    private LogRepository logRepository;

    @Test
    public void test() {
        Log strLog = new Log();
        strLog.setData("string here");
        logRepository.save(strLog);
        Log objLog = new Log();
        objLog.setData(new JsonObject().put("key", "value").put("obj", new JsonObject()));
        logRepository.save(objLog);
        Log aryLog = new Log();
        aryLog.setData(new JsonArray().add("a").add("b").add("c"));
        logRepository.save(aryLog);
    }
}

结果如下:

{
        "_id" : ObjectId("5a09fa46a15b065268a0a157"),
        "_class" : "ltd.linkcon.spider.domain.Log",
        "data" : "string here"
}
{
        "_id" : ObjectId("5a09fa46a15b065268a0a158"),
        "_class" : "ltd.linkcon.spider.domain.Log",
        "data" : {
                "key" : "value",
                "obj" : [ ]
        }
}
{
        "_id" : ObjectId("5a09fa46a15b065268a0a159"),
        "_class" : "ltd.linkcon.spider.domain.Log",
        "data" : [
                "a",
                "b",
                "c"
        ]
}

这个解决方案还可以避免在映射过程中Spring添加_classmap字段。 - ryanhex53

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接