web-dev-qa-db-ja.com

MongoDBを使用する方法は、 `$ lookup`を` findOne() `として集約します

あなたが知っているように、find()は結果の配列を返し、findOne()は単なるオブジェクトを返します。

Angularでは、これにより大きな違いが生じます。 {{myresult[0].name}}の代わりに、単に{{myresult.name}}と書くだけです。

集約パイプラインの$lookupメソッドは、単一のオブジェクトではなく、結果の配列を返すことがわかりました。

たとえば、2つのコレクションがあります。

usersコレクション:

[{
  "firstName": "John",
  "lastName": "Smith",
  "country": 123
}, {
  "firstName": "Luke",
  "lastName": "Jones",
  "country": 321
}]

countriesコレクション:

[{
  "name": "Australia",
  "code": "AU",
  "_id": 123
}, {
  "name": "New Zealand",
  "code": "NZ",
  "_id": 321
}]

マイ集計$lookup

db.users.aggregate([{
  $project: {
    "fullName": {
      $concat: ["$firstName", " ", "$lastName"]
    },
    "country": "$country"
  }
}, {
  $lookup: {
    from: "countries",
    localField: "country",
    foreignField: "_id",
    as: "country"
  }
}])

クエリの結果:

[{
  "fullName": "John Smith",
  "country": [{
    "name": "Australia",
    "code": "AU",
    "_id": 123
  }]
}, {
 "fullName": "Luke Jones",
 "country": [{
   "name": "New Zealand",
   "code": "NZ",
   "_id": 321
 }]
}]

上記の結果からわかるように、各country"country": {....}のような単一のオブジェクトではなく配列です。

$lookupは単一のドキュメントにのみ一致するため、配列ではなく単一のオブジェクトを返すようにするにはどうすればよいですか?

28
Fizzix

もうすぐそこにいるので、別の $project パイプラインへのステージングおよび $arrayElemAt は、配列内の単一の要素を返します。

db.users.aggregate(
    [
        {   "$project": {     
            "fullName": {       
                "$concat": [ "$firstName", " ", "$lastName"]     
            },
            "country": "$country"   
        }}, 
        { "$lookup": {     
                "from": "countries",     
                "localField": "country",     
                "foreignField": "_id",     
                "as": "countryInfo"   
        }}, 
        { "$project": { 
            "fullName": 1, 
            "country": 1, 
            "countryInfo": { "$arrayElemAt": [ "$countryInfo", 0 ] } 
        }} 
    ]
)
44
styvane

"preserveNullAndEmptyArrays"を使用することもできます

そのようです:

 db.users.aggregate(
        [
            {   "$project": {     
                "fullName": {       
                    "$concat": [ "$firstName", " ", "$lastName"]     
                },
                "country": "$country"   
            }}, 
            { "$lookup": {     
                    "from": "countries",     
                    "localField": "country",     
                    "foreignField": "_id",     
                    "as": "countryInfo"   
            }}, 
            {"$unwind": {
                    "path": "$countryInfo",
                    "preserveNullAndEmptyArrays": true
                }
            },
        ]
    )
8
Ros