あなたが知っているように、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
は単一のドキュメントにのみ一致するため、配列ではなく単一のオブジェクトを返すようにするにはどうすればよいですか?
もうすぐそこにいるので、別の $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 ] }
}}
]
)
"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
}
},
]
)