mongo數(shù)據集合屬性中存在點號(.)的解決方法
前言
MongoDB是面向集合存儲的文檔型數(shù)據庫,其涉及到的基本概念與關系型數(shù)據庫比有所不同。本文主要介紹關于mongo數(shù)據集合屬性存在點號(.)的相關內容,下面話不多說了,來一起看看詳細的介紹吧
基本知識點:
1.似乎mongo3.6之前不允許插入帶點(.)或美元符號($)的鍵,但是當我使用mongoimport工具導入包含點的JSON文件時,它工作正常。
2.在使用spring-data-mongodb處理mongodb的增刪改查時會通過一個MappingMongoConverter(Document和Modle轉換類)轉換數(shù)據
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ù)據庫中的樣子:
{
"_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類復制代碼自定義修改,編譯之后或優(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?Atlas?實現(xiàn)語義搜索與?RAG(邁向AI的搜索機制)
MongoDBAtlas的語義搜索功能通過向量化存儲和檢索非結構化數(shù)據,結合RAG框架,實現(xiàn)了高效的知識檢索和增強型生成應用,本文給大家介紹如何通過MongoDB?Atlas?實現(xiàn)語義搜索與?RAG(邁向AI的搜索機制),感興趣的朋友跟隨小編一起看看吧2024-11-11
mongodb使用docker搭建replicaSet集群與變更監(jiān)聽(最新推薦)
replicaSet和cluster從部署難度相比,replicaSet要簡單許多。如果所存儲的數(shù)據量規(guī)模不算太大的情況下,那么使用replicaSet方式部署mongodb是一個不錯的選擇,這篇文章主要介紹了mongodb使用docker搭建replicaSet集群與變更監(jiān)聽,需要的朋友可以參考下2023-03-03
MongoDB模糊查詢操作案例詳解(類關系型數(shù)據庫的 like 和 not like)
這篇文章主要介紹了MongoDB的模糊查詢操作(類關系型數(shù)據庫的 like 和 not like) ,本文通過代碼案例分析給大家介紹的非常詳細,具有一定的參考借鑒價值,,需要的朋友可以參考下2019-07-07

