mongo數(shù)據(jù)集合屬性中存在點號(.)的解決方法
前言
MongoDB是面向集合存儲的文檔型數(shù)據(jù)庫,其涉及到的基本概念與關系型數(shù)據(jù)庫比有所不同。本文主要介紹關于mongo數(shù)據(jù)集合屬性存在點號(.)的相關內容,下面話不多說了,來一起看看詳細的介紹吧
基本知識點:
1.似乎mongo3.6之前不允許插入帶點(.)或美元符號($)的鍵,但是當我使用mongoimport工具導入包含點的JSON文件時,它工作正常。
2.在使用spring-data-mongodb處理mongodb的增刪改查時會通過一個MappingMongoConverter(Document和Modle轉換類)轉換數(shù)據(jù)
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的屬性
問題描述:文檔在數(shù)據(jù)庫中的樣子:
{ "_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并沒有支持,但是因為一開始項目已經(jīng)使用了spring-data-mongodb難以替換,所以就想到覆蓋轉換方法。
怎么覆蓋spring-data-mongodb包中的文件?
新建一個和DBObjectAccessor轉換文件一樣的目錄,重新建DBObjectAccessor類復制代碼自定義修改,編譯之后或優(yōu)先使用新建的類。
//查詢時轉換 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本不支持點號,插入會報錯
當然最好不要發(fā)生屬性中有點號的情況。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關文章
MongoDB數(shù)據(jù)庫性能監(jiān)控詳解
MongoDB作為圖片和文檔的存儲數(shù)據(jù)庫,為啥不直接存MySQL里,還要搭個MongoDB集群,麻不麻煩?這篇文章就帶你介紹MongoDB數(shù)據(jù)庫性能監(jiān)控,感興趣的同學可以參考閱讀2023-03-03MongoDB的聚合框架Aggregation Framework入門學習教程
MongoDB中的聚合aggregate主要用于處理數(shù)據(jù)計算,這里我們就來詳細整理MongoDB的聚合框架Aggregation Framework入門學習教程,需要的朋友可以參考下2016-07-07MongoDB連接和創(chuàng)建數(shù)據(jù)庫的方法講解
這篇文章介紹了MongoDB連接和創(chuàng)建數(shù)據(jù)庫的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-03-03Linux系統(tǒng)下MongoDB的簡單安裝與基本操作
這篇文章主要介紹了Linux系統(tǒng)下MongoDB的簡單安裝與基本操作,需要的朋友可以參考下2015-04-04MongoDB的mongo shell常用操作方法及操作腳本筆記
mongo shell即相當于SQL語句在關系型數(shù)據(jù)庫中的作用,MongoDB使用JavaScript作為shell操作命令,這里我們就來整理MongoDB的mongo shell常用操作方法及操作腳本筆記2016-07-07關于CentOS 8 搭建MongoDB4.4分片集群的問題
在MongoDB里面存在另一種集群,就是分片技術,可以滿足MongoDB數(shù)據(jù)量大量增長的需求。這篇文章主要介紹了CentOS 8 搭建MongoDB4.4分片集群的問題,需要的朋友可以參考下2021-10-10解決mongodb在ubuntu下啟動失敗,提示couldn‘t remove fs lock errno:9 Bad
之前MongoDB啟動的時候是蠻正常的,不知道后來啟動報錯了,就把粘貼出來查詢了。經(jīng)過一番百度,才找的處理的辦法,分享給大家2014-08-08