理解MongoDB Explain

28

我执行了一个查询,尝试在Mongo控制台上解释它,得到了:

"isMultiKey" : true,
"n" : 8,
"nscannedObjects" : 17272,
"nscanned" : 17272,
"nscannedObjectsAllPlans" : 21836,
"nscannedAllPlans" : 21836,
"scanAndOrder" : true,
"indexOnly" : false,
"nYields" : 0,
"nChunkSkips" : 0,
"millis" : 184,

大部分问题在http://www.mongodb.org/display/DOCS/Explain中都有解释,但我不太理解nscannedObjectsAllPlans和nscannedAllPlans是什么意思。有人能帮忙解答吗?

谢谢。

1个回答

24

nscannednscannedObjects报告的是获胜的查询计划的结果。

nscannedAllPlansnscannedObjectsAllPlans报告的是所有计划的结果。

文档

索引条目扫描次数。 totalKeysExamined对应于MongoDB早期版本中cursor.explain()返回的nscanned字段。

例如:

t = db.jstests_explainb;
t.drop();

t.ensureIndex( { a:1, b:1 } );
t.ensureIndex( { b:1, a:1 } );

t.save( { a:0, b:1 } );
t.save( { a:1, b:0 } );

// Older mongodb (< 3.0? )
t.find( { a:{ $gte:0 }, b:{ $gte:0 } } ).explain( true );
    {
      "isMultiKey": false,
      "n": 2,
      "nscannedObjects": 2,
      "nscanned": 2,
      "nscannedObjectsAllPlans": 6,
      "nscannedAllPlans": 6,
      "scanAndOrder": false,
      "indexOnly": false,
      "nYields": 0,
      "nChunkSkips": 0,
      "millis": 2,
    ...
    }

// MongoDB 4.4
t.find( { a:{ $gte:0 }, b:{ $gte:0 } } ).explain( true );
{
    "queryPlanner" : {
        "plannerVersion" : 1,
        "namespace" : "test.jstests_explainb",
        ...
        "queryHash" : "CB67518C",
        "planCacheKey" : "5E76CDD1",
        "winningPlan" : {
            "stage" : "FETCH",
            "inputStage" : {
                "stage" : "IXSCAN",
                "keyPattern" : {
                    "a" : 1,
                    "b" : 1
                },
                "indexName" : "a_1_b_1",
            }
        },
        "rejectedPlans" : [
            {
                "stage" : "FETCH",
                "inputStage" : {
                    "stage" : "IXSCAN",
                    "keyPattern" : {
                        "b" : 1,
                        "a" : 1
                    },
                    "indexName" : "b_1_a_1",
                }
            }
        ],
        ...
    },
    "executionStats" : {
        "executionSuccess" : true,
        "nReturned" : 2,
        "executionTimeMillis" : 0,
        "totalKeysExamined" : 2, // <-- same as `nscanned`
        "totalDocsExamined" : 2, // <--
        "executionStages" : { ... }
        "allPlansExecution" : [
            {...},
            {...}
        ]
    }


如果nscannedObjectsAllPlans和nscannedAllPlans的值很大,而nscanned仍在可接受的范围内,那么我应该担心吗? - Global Warrior
我不这么认为 - 我相信只有在调用 explain() 函数时,它才会首先执行那些其他的计划。 - Eric Mill
4
运行多个计划基本上是MongoDB决定使用哪个查询计划的方式。它在第一次运行查询时并行执行所有可能使用的计划,然后在一定数量的查询或数据更改之后定期执行。一旦一个计划完成,它将其缓存为要使用的计划,取消其他计划,并将使用该计划直到下一次决定刷新它。 - David Mason
@DavidMason 查询优化器根据哪个计划最快地返回101个文档来决定使用哪个计划。请参见此处http://docs.mongodb.org/manual/core/query-plans/。 - Gianfranco P.

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接