众所周知,
find()
返回一个结果数组,而findOne()
仅返回一个简单的对象。
对于 Angular,这会产生巨大的差异。我可以简单地写
{{myresult[0].name}}
,而不是去 {{myresult.name}}
。
我发现聚合管道中的
$lookup
方法返回结果数组,而不仅仅是单个对象。
例如,我有两个收藏:
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
}
},
]
)
db.users.aggregate([
{
$lookup: {
from: 'countries',
localField: 'country',
foreignField: '_id',
as: 'country'
}
},
{
$unwind: '$country'
}
]).pretty()
您可以使用此 mongo 查询来获取国家/地区对象
当您不想重复项目中的所有字段时,只需使用 $addFields 覆盖有问题的字段即可:
db.users.aggregate([
{ "$project": {
"fullName": {
"$concat": [ "$firstName", " ", "$lastName"]
},
"country": "$country"
}},
{ "$lookup": {
"from": "countries",
"localField": "country",
"foreignField": "_id",
"as": "countryInfo"
}},
{ "$addFields": {
"countryInfo": {
"$arrayElemAt": [ "$countryInfo", 0 ]
}
}}
])
我认为最干净的方法是在管道中添加另一个步骤:
...,
{
$addFields : {
$country: { $first: '$country' }
}
}
您可以使用 $project 或 $addFields。
我们可以使用 $project 的地方是,如果您知道进一步需要的所有字段是什么,在这种情况下您可以使用 $project。
示例:
{
"$project": {
"fullName": 1,
"country": 1,
"countryInfo": { "$arrayElemAt": [ "$countryInfo", 0 ] }
}
}
我们可以使用 $addFields 的地方是,如果您不确定其余字段,而您只想修改特定的查找字段,在这种情况下,我们可以使用 $addFields。
示例。
{
"$addFields": {
"countryInfo": {
"$arrayElemAt": ["$countryInfo", 0]
}
}
}