使用“$ arrayElemAt”返回一个元素

时间:2016-12-09 10:52:50

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

如果我在这里混淆了一些术语,请原谅我,但我正在使用'$ 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();
});

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

2 个答案:

答案 0 :(得分:6)

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

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

答案 1 :(得分:0)

看来这是 still ,截至2018年8月,实现将通过$lookup连接到单个嵌入式文档并过滤其字段的文档投影的最佳方法仍然是通过辅助$project流水线阶段。尽管@styvane上面的 方法确实可行,并且确实很新颖,但它却增加了管道的复杂性,因此我很犹豫地接受此作为答案。