2010-02-15 110 views
15

假設我有一個BlogPost模型與零對多嵌入式Comment文件。我可以查詢並讓MongoDB返回只有Comment符合我的查詢規格的對象?MongoDB查詢返回僅嵌入文檔

例如,db.blog_posts.find({"comment.submitter": "some_name"})只返回評論列表。

編輯:一個例子:

import pymongo 

connection = pymongo.Connection() 
db = connection['dvds'] 

db['dvds'].insert({'title': "The Hitchhikers Guide to the Galaxy", 
        'episodes': [{'title': "Episode 1", 'desc': "..."}, 
           {'title': "Episode 2", 'desc': "..."}, 
           {'title': "Episode 3", 'desc': "..."}, 
           {'title': "Episode 4", 'desc': "..."}, 
           {'title': "Episode 5", 'desc': "..."}, 
           {'title': "Episode 6", 'desc': "..."}]}) 

episode = db['dvds'].find_one({'episodes.title': "Episode 1"}, 
           fields=['episodes']) 
在這個例子中

episode是:

{u'_id': ObjectId('...'), 
u'episodes': [{u'desc': u'...', u'title': u'Episode 1'}, 
       {u'desc': u'...', u'title': u'Episode 2'}, 
       {u'desc': u'...', u'title': u'Episode 3'}, 
       {u'desc': u'...', u'title': u'Episode 4'}, 
       {u'desc': u'...', u'title': u'Episode 5'}, 
       {u'desc': u'...', u'title': u'Episode 6'}]} 

,但我只是想:

{u'desc': u'...', u'title': u'Episode 1'} 
+0

已接受答案的「未來」現在已成爲過去,因爲聚合框架已經存在。我回答瞭如何去做。 – 2013-08-31 12:27:13

回答

2

JavaScript的外殼在http://docs.mongodb.org/manual/reference/method/記錄MongoDB的

如果你想找回只有一個對象的特定字段,可以使用

db.collection.find({ }, {fieldName:true}); 

如果,另一方面,你正在尋找含有特定字段的對象,你可以起訴

db.collection.find({ fieldName : { $exists : true } }); 
+0

我試圖用原始帖子中的示例來澄清我的問題。謝謝。 – Carson 2010-02-15 20:43:29

0

db.eval

你應該這樣做:

episode = connection['dvds'].eval('function(title){ 
    var t = db.dvds.findOne({"episodes.title" : title},{episodes:true}); 
    if (!t) return null; 
    for (var i in t.episodes) if (t.episodes[i].title == title) return t.episodes[i]; 
}', "Episode 1"); 

所以情節的過濾會在服務器端。

7

我想你想的是:

print db.dvds.aggregate([ 
    {"$unwind": "$episodes"}, # One document per episode 
    {"$match": {"episodes.title": "Episode 1"} }, # Selects (filters) 
    {"$group": {"_id": "$_id", # Put documents together again 
       "episodes": {"$push": "$episodes"}, 
       "title": {"$first": "$title"} # Just take any title 
      } 
    }, 
])["result"] 

輸出(除了空格)是:

[ { u'episodes': [ { u'title': u'Episode 1', 
        u'desc': u'...' 
        } 
       ], 
    u'_id': ObjectId('51542645a0c6dc4da77a65b6'), 
    u'title': u'The Hitchhikers Guide to the Galaxy' 
    } 
] 

如果你想獲得從u"_id"中刪除,追加管道:

{"$project": {"_id": 0, 
       "episodes": "$episodes", 
       "title": "$title"} 
       } 
0

我也遇到了同樣的問題。 我的方式是使用聚合函數。首先展開它然後匹配它。

db.dvds.aggregate([{$unwind:"$episodes"},{$match:{"episodes.title":"Episode 1"}}]) 

結果會像

{ "_id" : ObjectId("5a129c9e6944555b122c8511"), 
    "title" : "The Hitchhikers Guide to the Galaxy", 
    "episodes" : { "title" : "Episode 1", "desc" : "..." } } 

它並不完美,但你可以通過蟒蛇編輯。