美文网首页我爱编程
MongoDB查询优化器

MongoDB查询优化器

作者: cb9e58ff5a37 | 来源:发表于2017-12-20 15:44 被阅读0次

    索引概述

    介绍查询优化器首先要从索引开始。索引在计算机系统中应用非常广泛,是提高查询效率的常用手段。如果没有索引,MongoDB必须遍历集合中所有文档才能找到匹配的结果;如果存在一个适当的索引可以限制MongoDB必须检查的文档数量。

    在MongoDB中,索引是一种特殊的数据结构,以一种便于遍历的方式存储集合数据的部分信息。

    常见的索引有几种组织模型,其中,B-Tree索引可以看做将键值映射到有序数组中的位置;Hash索引将键值映射到无序数组中的位置。MongoDB默认采用B-Tree组织索引文件,在2.4版本后也允许建立Hash索引。

    索引的创建

    MongoDB索引是在集合上创建的,通过ensureIndex方法可以方便的设置索引。MongoDB集合可以创建多个索引。恰当的索引可以显著提高查询效率,但是当数据发生变动时,不仅需要修改更新文档,还要更新集合上的索引,这无疑会导致写操作要花费更多的时间。因此,MongoDB限制每个集合最多可以有64个索引,当然,只要设计合理,这个上限也是足够的。
    我们用《MongoDB 权威指南》中的例子做一下修改,假设用MongoDB存储百万级用户数据,包括姓名(name), 性别(sex), 年龄(age),电话(tel)信息。为了便于说明问题,这里只写四条数据:

    > db.users.find()
    { "_id" : ObjectId("5a2f8e7bd24b2183ae3b9fab"), "name" : "Andy", "sex" : "male", "age" : 16, "tel" : "1300000001" }
    { "_id" : ObjectId("5a2f97c7eb221206e13c7c3f"), "name" : "Bale", "sex" : "male", "age" : 18, "tel" : "1300000002" }
    { "_id" : ObjectId("5a2fa15beb221206e13c7c40"), "name" : "Cindy", "sex" : "female", "age" : 17, "tel" : "1300000003" }
    { "_id" : ObjectId("5a2fa1cbeb221206e13c7c41"), "name" : "Dany", "sex" : "male", "age" : 20, "tel" : "1300000004" }
    

    我们创建三个索引:

    > db.users.ensureIndex({"age": 1, "sex": 1});
    {
        "createdCollectionAutomatically" : false,
        "numIndexesBefore" : 1,
        "numIndexesAfter" : 2,
        "ok" : 1
    }
    > db.users.ensureIndex({"age": 1});
    {
        "createdCollectionAutomatically" : false,
        "numIndexesBefore" : 2,
        "numIndexesAfter" : 3,
        "ok" : 1
    }
    > db.users.ensureIndex({"sex": 1, "age": 1});
    {
        "createdCollectionAutomatically" : false,
        "numIndexesBefore" : 3,
        "numIndexesAfter" : 4,
        "ok" : 1
    }
    

    在创建集合时,MongoDB会默认在_id字段上创建一个唯一索引,并且不能删除。所以当我们创建第一个索引时numIndexesBefore值是1。需要注意的是,B-Tree索引是有序的,上面三个索引大致结构是:

    # {"age": 1, "sex": 1}
    [16, male]      ->  ObjectId("5a2f8e7bd24b2183ae3b9fab")
    [17, female]    ->  ObjectId("5a2fa15beb221206e13c7c40")
    [18, male]      ->  ObjectId("5a2f97c7eb221206e13c7c3f")
    [20, male]      ->  ObjectId("5a2fa1cbeb221206e13c7c41")
    
    # {"age": 1}
    [16]      ->  ObjectId("5a2f8e7bd24b2183ae3b9fab")
    [17]      ->  ObjectId("5a2fa15beb221206e13c7c40")
    [18]      ->  ObjectId("5a2f97c7eb221206e13c7c3f")
    [20]      ->  ObjectId("5a2fa1cbeb221206e13c7c41")
    
    # {"sex": 1, "age": 1}
    [female, 17]    ->  ObjectId("5a2fa15beb221206e13c7c40")
    [male, 16]      ->  ObjectId("5a2f8e7bd24b2183ae3b9fab")
    [male, 18]      ->  ObjectId("5a2f97c7eb221206e13c7c3f")
    [male, 20]      ->  ObjectId("5a2fa1cbeb221206e13c7c41")
    

    索引的选择

    MongoDB一个集合可以创建多个索引,查询优化器为索引创建查询计划,将选择索引转换为选择查询计划或者说选择查询计划对应的游标。

    由于最近版本中几乎看不到优化器的概念,我们从较早的版本中开始介绍,以MongoDB-2.4.14为例,具体的选择过程可以分为三个阶段:首先根据查询模式(Query pattern)判断是否存在CachedPlan,如果存在直接选择,这里查询模式是除了数值外其他查询条件完全相同;其次,如果没有缓存记录,查询优化器创建新查询计划并标记类型,如果类型为Optimal Plan则直接执行该Plan;最后,如果不存在Optimal Plan,MongoDB会并发尝试可能的Helpful Plan以及不使用索引的基础查询。查询优化器会对比选择表现最好的查询计划继续执行,并将查询模式与最终查询计划的映射写入CachedPlan。

    Optimal Plan

    Optimal Plan的索引首先应该包含所有查询的过滤字段和排序字段,其次,字段在索引中的顺序是范围过滤和排序字段位于相等字段之后。有多个Optimal Plan时,会执行第一个。

    Helpful Plan

    不存在Optimal Plan时,查询优化器会并发执行所有Helpful查询计划。最早得到101个结果的查询计划会被选中继续执行,并将该查询计划暂存。其他查询计划会被中止。

    下面列举几个例子,为了便于说明,我们将集合文档数扩大到1000000:

    > var names = ["Andy", "Bale", "Cindy", "Dany", "Emmy", "Ford", "Gill"]
    > var sexs = ["male", "female"]
    > for (i=0; i<1000000; i++) {
        db.users.insert(
            {"name": names[Math.floor(Math.random()*7)],
             "sex": sexs[Math.floor(Math.random()*2)],
             "age": Math.floor(Math.random()*100),
             "tel": 13000000001+i
            } 
        ); 
    }
    

    首先查询age为18的所有记录,这里只截取部分输出信息:

    > db.users.find({"age": 18}).explain({"verbose": true})
    {
        "cursor" : "BtreeCursor age_1_sex_1",
        "isMultiKey" : false,
        "n" : 9777,
        "nscannedObjects" : 9777,
        "nscanned" : 9777,
        "millis" : 2168,
        "allPlans" : [
            {
                "cursor" : "BtreeCursor age_1_sex_1",
                "n" : 9777,
                "nscannedObjects" : 9777,
                "nscanned" : 9777,
            }
        ],
    }
    

    根据定义“age_1_sex_1”和“age_1”对应的查询计划都是Optimal Plan,这里选取了第一个便不再查找其他Plan,所以allPlans也只有“age_1_sex_1”一个。
    下面第二次查询age为18的记录,同样截取部分输出:

    > db.users.find({"age": 18}).explain({"verbose": true})
    {
        "cursor" : "BtreeCursor age_1_sex_1",
        "isMultiKey" : false,
        "n" : 9777,
        "nscannedObjects" : 9777,
        "nscanned" : 9777,
        "millis" : 1686,
        "allPlans" : [
            {
                "cursor" : "BtreeCursor age_1_sex_1",
                "n" : 9777,
                "nscannedObjects" : 9777,
                "nscanned" : 9777,
            }
        ],
        "oldPlan" : {
            "cursor" : "BtreeCursor age_1_sex_1",
        },
    }
    

    与第一次不同,这里多了oldPlan。注意:

    • 根据查询模式查找缓存查询计划,此时如果查找age为20只有值不同,同样会命中缓存。
    • 由于指定了explain(),本次查询并没有直接使用缓存计划而是需要重新评估。

    缓存的查询计划在以下条件下会清空并重新评估:

    • 集合收到1000次写操作
    • 执行reindex
    • 添加或删除索引
    • mongod进程重启
    • 查询时指定explain()

    第三个例子,我们查询name为Andy,age为18的记录:

    > db.users.find({"age": 18, "name": "Andy"}).explain({"verbose": true})
    {
        "cursor" : "BtreeCursor age_1",
        "isMultiKey" : false,
        "n" : 1392,
        "nscannedObjects" : 9777,
        "nscanned" : 9777,
        "millis" : 889,
        "allPlans" : [
            {
                "cursor" : "BtreeCursor age_1",
                "n" : 1348,
                "nscannedObjects" : 9777,
                "nscanned" : 9777,
            },
            {
                "cursor" : "BtreeCursor age_1_sex_1",
                "n" : 91,
                "nscannedObjects" : 700,
                "nscanned" : 700,
            },
            {
                "cursor" : "BasicCursor",
                "n" : 0,
                "nscannedObjects" : 700,
                "nscanned" : 700,
    
            }
        ],
    }
    

    此次查询没有最优索引,两个索引“age_1_sex_1”和“age_1”对查询有帮助,连同基础查询总共创建三个查询计划。
    评估候选查询计划会在以下情况之一发生时结束:

    • 获取到所有查询结果;
    • 最先得到101个查询结果(enoughCumulativeMatchesToChooseAPlan)
      上面例子中,“age_1”最先返回101个查询结果,对应的查询计划执行到最后,其他查询计划被中止。

    在MongoDB较新的版本中(以3.2.10为例),查询优化器做了很大的改变。对于每个查询,系统首先在查询计划缓存中匹配相同查询信息(Query Shape)选择查询计划;如果没有命中缓存,查询计划生成器会评估所有候选查询计划,选择一个最优计划(Winning Plan),并将最有计划写入缓存。
    当缓存命中时,系统会重新评估该缓存计划,作出是否合格的判定,并根据判定结果保留或清除缓存计划。
    下面是3.2.10版本的一次查询:

    replset:PRIMARY> db.users.find({"age": 18, "sex": "male"}).explain("queryPlanner")
    {
        "queryPlanner" : {
            "namespace" : "date.users",
            "indexFilterSet" : false,
            "winningPlan" : {
                "stage" : "FETCH",
                "inputStage" : {
                    "indexName" : "sex_1_age_1",
                }
            },
            "rejectedPlans" : [
                {
                    "inputStage" : {
                        "indexName" : "age_1",
                    }
                },
                {
                    "inputStage" : {
                        "indexName" : "age_1_sex_1",
                    }
                }
            ]
        },
    }
    

    两个版本最主要的区别就是新版本取消了直接通过规则选择Optimal Plan的逻辑,按照以前的匹配规则,有“sex_1_age_1”这样的Optimal Plan不会再执行其他Optimal Plan和Helpfun Plan。取消Optimal Pan是因为往往Optimal Plan并不是最优的。下面是MongoDB-2.4.14的两个例子,同样只截取部分信息说明:

    > db.users.find({"age": 18}, {"age": 1, "sex": 1, "_id": 0}).explain("verbose")
    {
        "cursor" : "BtreeCursor age_1",
        "n" : 9777,
        "nscannedObjects" : 9777,
        "nscanned" : 9777,
        "indexOnly" : false,
        "millis" : 755,
        "allPlans" : [
            {
                "cursor" : "BtreeCursor age_1",
                "n" : 9777,
                "nscannedObjects" : 9777,
                "nscanned" : 9777,
            }
        ],
    }
    > db.users.find({"age": 18}, {"age": 1, "sex": 1, "_id": 0}).hint("age_1_sex_1").explain("verbose")
    {
        "cursor" : "BtreeCursor age_1_sex_1",
        "n" : 9777,
        "nscannedObjects" : 0,
        "indexOnly" : true,
        "millis" : 27,
        "allPlans" : [
            {
                "cursor" : "BtreeCursor age_1_sex_1",
                "n" : 9777,
                "nscannedObjects" : 0,
                "nscanned" : 9777,
            }
        ],
    }
    
    

    上面的两次查询中,第一次查询“age_1”首先匹配到Optimal Plan并执行到最后,用时755毫秒,“age_1_sex_1”虽然同样是Optimal Plan,但是索引的顺序排在“age_1”之后,所以没有被选中。第二次查询中通过hint显式指定“age_1_sex_1”,通过该索引查询是“indexOnly”,不需要扫描文档就可以完成查询,所以效率更高,用时只有27毫秒。

    另外,查询优化器还有一个变化,匹配缓存计划由Query Pattern变为Query Shape。官网上对二者的定义分别是:

    • Query pattern refers to query select conditions that differ only in the values.
    • A query shape consists of a combination of query, sort, and projection specifications.

    Query Pattern 关注的是Fileds,Range和Sort,而Query Shape增加了Projection。

    总结

    查询优化器极大提升了查询性能,随着版本的升级,虽然最新的代码中已经很难找到“optimizer”的字段,但是逻辑仍然保留并且变得更加完善。
    目前,查询优化器并不能保证每次查询都是最优的,无论是OptimalPlan 还是WinningPlan都是相对的,配合hint,index filter会使查询优化器更精准。
    查询优化器是一次MongoDB查询的一个重要环节,深入了解可以帮助我们更合理的设计索引,更高效的使用MongoDB。

    相关文章

      网友评论

        本文标题:MongoDB查询优化器

        本文链接:https://www.haomeiwen.com/subject/ovbgwxtx.html