在mongodb中的聚合管道中展开操作后,查找阶段不起作用

时间:2020-06-28 16:06:18

标签: mongodb flask

在mongodb中相对较新时,我用这三个表执行了一个聚合管道,并且令人惊讶的是,在管道的第三个查找阶段中,我将“ r2”数组设置为空。我交叉检查了字段名一切都很好。

db.user.aggregate([
{
    $lookup:{
            from:"enrollment",
            localField:"user_id",
            foreignField:"userID",
            as:"r1"
        }
},

{
    $unwind:{
            path:"$r1",
            includeArrayIndex:"r1_id"
        }
},
{
    $lookup:{
            from:"course",
            localField:"r1.courseID",
            foreignField:"courseID",
            as:"r2"
        }
}
])

我有三个集合,分别是 user,注册和课程

用户身份

> db.user.find()
{ "_id" : ObjectId("5ef4ba8d500ac8876da0d2ca"), "user_id" : 1, "first_name" : "Christian", 
"last_name" : "Hur", "email" : "christian@uta.com", "password" : "abc1234" }
{ "_id" : ObjectId("5ef4ba8d500ac8876da0d2cb"), "user_id" : 2, "first_name" : "Mary", "last_name" : 
"Jane", "email" : "mary.jane@uta.com", "password" : "password123" }
{ "_id" : ObjectId("5ef4bc2563742adee5403b1d"), "user_id" : 3, "first_name" : "ari", "last_name" : 
"dutta", "email" : "dutta@uta.com", "password" : "po1234" }

然后,当然是

> db.course.find()
{ "_id" : ObjectId("5ef4c1b64a77aec0af5e73ae"), "courseID" : 3333, "title" : "Adv PHP 201", 
"description" : "Advance PHP programming", "credits" : 3, "term" : "fall" }
{ "_id" : ObjectId("5ef4c20d4a77aec0af5e73af"), "courseID" : 5555, "title" : "Java 201", 
"description" : "Advanced Programming", "credits" : 4, "term" : "fall"}
{ "_id" : ObjectId("5ef4c2564a77aec0af5e73b0"), "courseID" : 6666, "title" : "Angular 1", 
"description" : "Intro to Angular", "credits" : 3, "term" : "fall,spring" }

注册为

> db.enrollment.find()
{ "_id" : ObjectId("5ef771f42d98ffab4460a651"), "userID" : 1, "courseID" : "3333" }
{ "_id" : ObjectId("5ef7722d2d98ffab4460a652"), "userID" : 1, "courseID" : "6666" }

结果

{ "_id" : ObjectId("5ef4ba8d500ac8876da0d2ca"), "user_id" : 1, "first_name" : 
"Christian", "last_name" : "Hur", "email" : "christian@uta.com", "password" : 
"abc1234", "r1" : { "_id" : ObjectId("5ef771f42d98ffab4460a651"), "userID" : 
1, "courseID" : "3333" }, "r1_id" : NumberLong(0), "r2" : [ ] }

{ "_id" : ObjectId("5ef4ba8d500ac8876da0d2ca"), "user_id" : 1, "first_name" : 
"Christian", "last_name" : "Hur", "email" : "christian@uta.com", "password" : 
"abc1234", "r1" : { "_id" : ObjectId("5ef7722d2d98ffab4460a652"), "userID" : 
1, "courseID" : "6666" }, "r1_id" : NumberLong(1), "r2" : [ ] }

我也检查了文档,但没有找到帮助,我该如何解决?

1 个答案:

答案 0 :(得分:1)

在两个集合之间的courseID上存在类型错误,在enrollment中,类型为string,在course中,类型为number

将您的$lookup更改为此:

  {
    $lookup: {
      from: "course",
      let: {
        courseID: {
          $toInt: "$r1.courseID"
        }
      },
      pipeline: [
        {
          $match: {
            $expr: {
              $eq: [
                "$$courseID",
                "$courseID"
              ]
            }
          }
        }
      ],
      as: "r2"
    }
  }

亲自尝试: Mongo Playground

此外,由于您是新手,因此我个人建议您使用_id字段,而不要使用自己生成的courseID / user_id。这样只会更易于维护。

对于Mongo 4.2+版本,以下是如何更新注册收集字段:

db.enrollment.updateMany(
    {},
    [
        {
            $set: {
                courseID: {$toInt: "$courseID"}
            }
        }
    ]
)