1. 程式人生 > 資料庫 >mongo資料集合屬性中存在點號(.)的解決方法

mongo資料集合屬性中存在點號(.)的解決方法

前言

MongoDB是面向集合儲存的文件型資料庫,其涉及到的基本概念與關係型資料庫比有所不同。本文主要介紹關於mongo資料集合屬性存在點號(.)的相關內容,下面話不多說了,來一起看看詳細的介紹吧

基本知識點:

1.似乎mongo3.6之前不允許插入帶點(.)或美元符號($)的鍵,但是當我使用mongoimport工具匯入包含點的JSON檔案時,它工作正常。

2.在使用spring-data-mongodb處理mongodb的增刪改查時會通過一個MappingMongoConverter(Document和Modle轉換類)轉換資料

3.具體對點號的轉換在DBObjectAccessor(spring-data-mongodb-1.10.13)或者DocumentAccessor(spring-data-mongodb-2.0.9),如下:

//插入時轉換
public void put(MongoPersistentProperty prop,Object value) {
 Assert.notNull(prop,"MongoPersistentProperty must not be null!");
 String fieldName = prop.getFieldName();
 if (!fieldName.contains(".")) {
  dbObject.put(fieldName,value);
  return;
 }
 Iterator<String> parts = Arrays.asList(fieldName.split("\\.")).iterator();
 DBObject dbObject = this.dbObject;
 while (parts.hasNext()) {
  String part = parts.next();
  if (parts.hasNext()) {
   dbObject = getOrCreateNestedDbObject(part,dbObject);
  } else {
   dbObject.put(part,value);
  }
 }
}

//查詢時轉換
public Object get(MongoPersistentProperty property) {
 String fieldName = property.getFieldName();
 if (!fieldName.contains(".")) {
  return this.dbObject.get(fieldName);
 }
 Iterator<String> parts = Arrays.asList(fieldName.split("\\.")).iterator();
 Map<String,Object> source = this.dbObject;
 Object result = null;
 while (source != null && parts.hasNext()) {
  result = source.get(parts.next());
  if (parts.hasNext()) {
   source = getAsMap(result);
  }
 }
 return result;
}

//判斷值是否為空
public boolean hasValue(MongoPersistentProperty property) {
 Assert.notNull(property,"Property must not be null!");
 String fieldName = property.getFieldName();
 if (!fieldName.contains(".")) {
  return this.dbObject.containsField(fieldName);
 }
 String[] parts = fieldName.split("\\.");
 Map<String,Object> source = this.dbObject;
 Object result = null;
 for (int i = 1; i < parts.length; i++) {
  result = source.get(parts[i - 1]);
  source = getAsMap(result);
  if (source == null) {
   return false;
  }
 }
 return source.containsKey(parts[parts.length - 1]);
}

4.點號在mongodb中有子集合的含義

例如查詢A.B屬性:查詢的是集合中A對應子集合中的屬性B的值,並不是查詢集合中A.B的屬性  

問題描述:文件在資料庫中的樣子:

{
 "_id": ObjectId("5bae00765500af6307755111"),"name": "java","age": 26,"A.B": "nnnn"
}

因此在Model中使用@Field("A.B")查詢不出集合中的"A.B"的值

@Field("A.B")
@JSONField(serialzeFeatures = SerializerFeature.DisableCircularReferenceDetect)
private Integer ab;  

5.解決方法:

查閱多方資料有以下幾點體會:點號在MongoDB中可以插入應該開始於3.6版本,官方文件雖然說可以支援點號,但是第三方驅動、spring-data-mongodb並沒有支援,但是因為一開始專案已經使用了spring-data-mongodb難以替換,所以就想到覆蓋轉換方法。

怎麼覆蓋spring-data-mongodb包中的檔案?

新建一個和DBObjectAccessor轉換檔案一樣的目錄,重新建DBObjectAccessor類複製程式碼自定義修改,編譯之後或優先使用新建的類。

//查詢時轉換
public Object get(MongoPersistentProperty property) {
 String fieldName = property.getFieldName();
 return this.dbObject.get(fieldName);
}
 
//判斷值是否為空
public boolean hasValue(MongoPersistentProperty property) {
 Assert.notNull(property,"Property must not be null!");
 String fieldName = property.getFieldName();
 return this.dbObject.containsField(fieldName);
}

注意:儘量不要修改put方法,應為低版本的MongoDB本不支援點號,插入會報錯

當然最好不要發生屬性中有點號的情況。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對我們的支援。