如何使用 MongoDB 聚合`$lookup`作为`findOne()`

问题描述 投票:0回答:6

众所周知,

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
返回单个对象而不是数组,因为它只会匹配单个文档?

mongodb mongodb-query aggregation-framework
6个回答
102
投票

您已经快完成了,您需要在管道中添加另一个

$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 ] } 
        }} 
    ]
)

38
投票

您还可以使用

"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
                }
            },
        ]
    )

15
投票
db.users.aggregate([
    {
        $lookup: {
            from: 'countries',
            localField: 'country',
            foreignField: '_id',
            as: 'country'
        }
    },
    {
        $unwind: '$country'
    }
]).pretty()

您可以使用此 mongo 查询来获取国家/地区对象


9
投票

当您不想重复项目中的所有字段时,只需使用 $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 ]
        }
    }}
])

0
投票

我认为最干净的方法是在管道中添加另一个步骤:

...,
{
  $addFields : {
    $country: { $first: '$country' }
  }
}

0
投票

您可以使用 $project$addFields

我们可以使用 $project 的地方是,如果您知道进一步需要的所有字段是什么,在这种情况下您可以使用 $project

示例:

{ 
    "$project": { 
        "fullName": 1, 
        "country": 1, 
        "countryInfo": { "$arrayElemAt": [ "$countryInfo", 0 ] } 
    }
} 

我们可以使用 $addFields 的地方是,如果您不确定其余字段,而您只想修改特定的查找字段,在这种情况下,我们可以使用 $addFields

示例。

{
    "$addFields": {
        "countryInfo": {
            "$arrayElemAt": ["$countryInfo", 0]
        }
    }
}
© www.soinside.com 2019 - 2024. All rights reserved.