mongodb中的Redact似乎对我来说很模糊

时间:2015-09-18 13:36:57

标签: mongodb aggregation-framework

我现在正在与编辑斗争,我不确定理解它。

我刚刚阅读了文档并试图在集合等级上使用redact(它来自mongodb在线培训)

集合“成绩”中的文档如下所示:

{
    "_id" : ObjectId("50b59cd75bed76f46522c34e"),
    "student_id" : 0,
    "class_id" : 2,
    "scores" : [ 
        {
            "type" : "exam",
            "score" : 57.92947112575566
        }, 
        {
            "type" : "quiz",
            "score" : 21.24542588206755
        }, 
        {
            "type" : "homework",
            "score" : 68.19567810587429
        }, 
        {
            "type" : "homework",
            "score" : 67.95019716560351
        }, 
        {
            "type" : "homework",
            "score" : 18.81037253352722
        }
    ]
}

我使用以下查询:

db.grades.aggregate([
    { $match: { student_id: 0 } },
    { 
        $redact: {
            $cond: {
                if: { $eq: [ "$type" , "exam" ] },
                then: "$$PRUNE",
                else: "$$DESCEND"
            }
        }
    }

]    );

使用此查询,找到每种类型的检查,应排除此子文档。它的工作原理是:

{
    "_id" : ObjectId("50b59cd75bed76f46522c34e"),
    "student_id" : 0,
    "class_id" : 2,
    "scores" : [ 
    {
        "type" : "quiz",
        "score" : 21.24542588206755
    }, 
    {
        "type" : "homework",
        "score" : 68.19567810587429
    }, 
    {
        "type" : "homework",
        "score" : 67.95019716560351
    }, 
    {
        "type" : "homework",
        "score" : 18.81037253352722
    }
]
}

但如果我改变了条件,我希望结果中只保留考试:

if: { $eq: [ "$type" , "exam" ] },
       then: "$$DESCEND",
       else: "$$PRUNE" 

然而结果是空的。

我不明白为什么不包括“考试”类型的子文档。

1 个答案:

答案 0 :(得分:9)

$redact阶段从根文档及其字段开始,并且只有当该文档满足$$DESCEND的条件时,它才会检查该文档中包含的子文档。这意味着$ redact对你的文档做的第一件事是检查这个:

{
    "_id" : ObjectId("50b59cd75bed76f46522c34e"),
    "student_id" : 0,
    "class_id" : 2,
    "scores" : [] // Some array. I will look at this later.
}

此处甚至找不到type字段,因此$eq: [ "$type" , "exam" ]为false。当条件为假时,你告诉$ redact做了什么? else: "$$PRUNE",因此在审查子文件之前整理整个文件。

作为解决方法,请测试$type"exam"还是不存在。你没有明确地要求一个有效的解决方案,所以我会把它作为练习让你知道如何做到这一点。