38

我有一组用户,其中每个文档都具有以下结构:

{
  "_id": "<id>",
  "login": "xxx",
  "solved": [
    {
      "problem": "<problemID>",
      "points": 10
    },
    ...
  ]
}

该字段solved可能为空或包含任意多个子文档。我的目标是获取用户列表以及总分(总和points),其中尚未解决任何问题的用户将被分配总分 0。这是否可以通过单个查询来做到这一点(理想情况下使用聚合框架)?

我试图在聚合框架中使用以下查询:

{ "$group": {
  "_id": "$_id",
  "login": { "$first": "$login" },
  "solved": { "$addToSet": { "points": 0 } }
} }
{ "$unwind": "$solved" }
{ "$group": {
  "_id": "$_id",
  "login": { "$first": "$login" },
  "solved": { "$sum": "$solved.points" }
} }

但是我收到以下错误:

exception: The top-level _id field is the only field currently supported for exclusion

先感谢您

4

3 回答 3

117

使用 MongoDB 3.2 和更高版本,$unwind操作员现在有一些选项,特别是该preserveNullAndEmptyArrays选项可以解决这个问题。

如果此选项设置为 true,并且路径为 null、缺失或空数组,$unwind则输出文档。如果为 false,$unwind则在路径为 null、缺失或空数组时不输出文档。在您的情况下,将其设置为 true:

db.collection.aggregate([
    { "$unwind": {
        "path": "$solved",
        "preserveNullAndEmptyArrays": true
    } },
    { "$group": {
        "_id": "$_id",
        "login": { "$first": "$login" },
        "solved": { "$sum": "$solved.points" }
    } }
])
于 2016-05-24T08:32:07.330 回答
8

这是解决方案 - 它假设“已解决”字段不存在、等于 null 或解决了一系列问题和分数。它不处理的情况是“解决”为空数组 - 尽管这将是您可以添加的简单附加调整。

project = {$project : {
        "s" : {
            "$ifNull" : [
                "$solved",
                [
                    {
                        "points" : 0
                    }
                ]
            ]
        },
        "login" : 1
    }
};
unwind={$unwind:"$s"};
group= { "$group" : {
        "_id" : "$_id",
        "login" : {
            "$first" : "$login"
        },
        "score" : {
            "$sum" : "$s.points"
        }
    }
}

db.students.aggregate( [ project, unwind, group ] );

于 2012-12-16T04:14:59.623 回答
0

$lookup 然后 $unwind 在查找数组中,这可能是空的


  let posts = await Post.aggregate<ActivityDoc>([
    {
      $match: {
        _id: new mongoose.Types.ObjectId(req.params.id),
      },
    },
    {
      $lookup: {
        from: 'users',
        localField: 'user',
        foreignField: '_id',
        as: 'user',
      },
    },
    {
      $unwind: '$user',
    },
    {
      $unwind: {
        path: '$user.follower',
        preserveNullAndEmptyArrays: true,
      },
    },
    {
      $match: {
        $or: [
          {
            $and: [
              {
                'privacy.mode': {
                  $eq: PrivacyMode.EveryOne,
                },
              },
            ],
          },
          {
            $and: [
              {
                'privacy.mode': {
                  $eq: PrivacyMode.MyCircle,
                },
              },
              {
                'user.follower.id': {
                  $eq: req.currentUser?.id,
                },
              },
            ],
          },
        ],
      },
    },
  ]);
于 2022-03-05T12:25:07.693 回答