前言
MongoDB是面向集合存儲(chǔ)的文檔型數(shù)據(jù)庫(kù),其涉及到的基本概念與關(guān)系型數(shù)據(jù)庫(kù)比有所不同。本文主要介紹關(guān)于mongo數(shù)據(jù)集合屬性存在點(diǎn)號(hào)(.)的相關(guān)內(nèi)容,下面話不多說(shuō)了,來(lái)一起看看詳細(xì)的介紹吧
基本知識(shí)點(diǎn):
1.似乎mongo3.6之前不允許插入帶點(diǎn)(.)或美元符號(hào)($)的鍵,但是當(dāng)我使用mongoimport工具導(dǎo)入包含點(diǎn)的JSON文件時(shí),它工作正常。
2.在使用spring-data-mongodb處理mongodb的增刪改查時(shí)會(huì)通過(guò)一個(gè)MappingMongoConverter(Document和Modle轉(zhuǎn)換類)轉(zhuǎn)換數(shù)據(jù)
3.具體對(duì)點(diǎn)號(hào)的轉(zhuǎn)換在DBObjectAccessor(spring-data-mongodb-1.10.13)或者DocumentAccessor(spring-data-mongodb-2.0.9),如下:
//插入時(shí)轉(zhuǎn)換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); } }}//查詢時(shí)轉(zhuǎn)換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.點(diǎn)號(hào)在mongodb中有子集合的含義
例如查詢A.B屬性:查詢的是集合中A對(duì)應(yīng)子集合中的屬性B的值,并不是查詢集合中A.B的屬性
問(wèn)題描述:文檔在數(shù)據(jù)庫(kù)中的樣子:
{ "_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.解決方法:
查閱多方資料有以下幾點(diǎn)體會(huì):點(diǎn)號(hào)在MongoDB中可以插入應(yīng)該開(kāi)始于3.6版本,官方文檔雖然說(shuō)可以支持點(diǎn)號(hào),但是第三方驅(qū)動(dòng)、spring-data-mongodb并沒(méi)有支持,但是因?yàn)橐婚_(kāi)始項(xiàng)目已經(jīng)使用了spring-data-mongodb難以替換,所以就想到覆蓋轉(zhuǎn)換方法。
怎么覆蓋spring-data-mongodb包中的文件?
新建一個(gè)和DBObjectAccessor轉(zhuǎn)換文件一樣的目錄,重新建DBObjectAccessor類復(fù)制代碼自定義修改,編譯之后或優(yōu)先使用新建的類。
//查詢時(shí)轉(zhuǎn)換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方法,應(yīng)為低版本的MongoDB本不支持點(diǎn)號(hào),插入會(huì)報(bào)錯(cuò)
當(dāng)然最好不要發(fā)生屬性中有點(diǎn)號(hào)的情況。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)VEVB武林網(wǎng)的支持。
新聞熱點(diǎn)
疑難解答
圖片精選