【问题标题】:Project an element returned with "$arrayElemAt"投影一个用“$arrayElemAt”返回的元素
【发布时间】:2017-04-24 20:00:33
【问题描述】:

如果我在这里混淆了一些术语,请原谅我,但我正在使用 '$lookup' 运算符在聚合中执行连接操作,如下所示:

db.collection('items').aggregate([{$match: {}},
{
    $lookup: {
        from: 'usr',
        localField: 'usr._id',
        foreignField: '_id',
        as: '__usr'
    }
}, {
    $project: {
        info: 1,
        timestamp: 1,
        usr: {
            "$arrayElemAt": [ "$__usr", 0 ]
        }
    }
}], (err, result) => {
    res.json(result);
    db.close();
});

我正在对聚合结果进行投影,并使用“$arrayElemAt”从结果数组中提取单个“usr”匹配项。 出于显而易见的原因,我不想返回包含敏感信息的整个 'usr' 记录。 我要做的是对使用“$arrayElemAt”操作返回的元素执行投影。 我能够做到这一点的唯一方法是使用原始投影的附加投影,如下所示:

db.collection('items').aggregate([{$match: {}},
{
    $lookup: {
        from: 'usr',
        localField: 'usr._id',
        foreignField: '_id',
        as: '__usr'
    }
}, {
    $project: {
        info: 1,
        timestamp: 1,
        usr: {
            "$arrayElemAt": [ "$__usr", 0 ]
        }
    }
}, {
    $project: {
        info: 1,
        timestamp: 1,
        usr: {
            "username": 1
        }
    }
}], (err, result) => {
    res.json(result);
    db.close();
});

有没有办法在没有重复投影的情况下实现这一点?

【问题讨论】:

    标签: node.js mongodb mongodb-query aggregation-framework


    【解决方案1】:

    您可以使用$let 表达式将$arrayElemAt 返回值分配给变量,并使用点符号访问in 表达式中的子文档字段。

    "usr": {
        "$let": {
            "vars": { 
                "field": { 
                    "$arrayElemAt": [ "$__usr", 0 ]
                }
            },
            "in": "$$field.username"
        }
    

    【讨论】:

      【解决方案2】:

      看起来这是 仍然,截至 2018 年 8 月,将通过 $lookup 加入的文档投影到单个嵌入式文档并过滤其字段的最佳方法仍然是通过二级$project流水线阶段。虽然上面@styvane 显示的方法确实有效,而且确实非常新颖,但它给管道增加了更多的复杂性,所以我不愿意接受这个作为答案。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2021-09-10
        • 1970-01-01
        • 1970-01-01
        • 2022-07-24
        • 1970-01-01
        • 1970-01-01
        • 2018-05-06
        相关资源
        最近更新 更多