mongodb的DBObject支援儲存多維陣列,在增加元素時使用"$push"操作符,在刪除元素時使用"$pull".
但是在做更新時問題就來了,mongodb首先支援使用"$"來定位元組中的某個元素,例如:
view plain
- > t.find()
- { "_id" : ObjectId("4b97e62bf1d8c7152c9ccb74"), "title" : "ABC",
- "comments" : [ { "by" : "joe", "votes" : 3 }, { "by" : "jane", "votes" : 7 } ] }
- > t.update( {'comments.by':'joe'}, {$inc:{'comments.$.votes':1}}, false, true )
- > t.find()
- { "_id" : ObjectId("4b97e62bf1d8c7152c9ccb74"), "title" : "ABC",
- "comments" : [ { "by" : "joe", "votes" : 4 }, { "by" : "jane", "votes" : 7 } ] }
但是"$"操作符僅支援一維數組的定位,在更新多維陣列時會返回錯誤資訊,例如有如下資料結構:
view plain
- {
- "_id":ObjectId("4b97e62bf1d8c7152c9ccb74"),
- ”comments“:[
- {
- "by":"joe",
- "votes":3,
- "replies":[
- {"by":"jane",
- "votes":2
- }]
- }]
- }
如果要將"replies"中{“by”:"jane"}的投票數增加1,該如何做呢(使用"$"mongodb會返回錯誤)?
mongodb使用多維陣列下標的方式來定位某個元素,那麼我們的方法就有了:
先find({"comments.replies.by":"jane"})來擷取到整個object,然後計算相應reply的數組下標,再使 用update({"comments.0.replies.0.by":"jane"},{"$inc", {"comments.0.replies.0.votes":1}}),即可,這樣就沒有同步的問題了。
ps:其實比較好的方法是由使用者用戶端程式來提供這個下標,即mongodb用戶端不要到伺服器擷取這個多維陣列再計算下標,效率太慢了。如果這個多維陣列已經呈現給使用者的話,那麼完全可以由使用者端的程式來提供這個下標,這樣效率是很高的。