MongoDB 复合索引在单独查询字段的时候不生效。

2015-12-26 14:52:18 +08:00
 1yndonn3u

数据如下:

> db.testcoll.find().limit(2)
{ "_id" : ObjectId("567ac8e9498f43a9cea8dd08"), "Name" : "User1", "Age" : 1, "Gender" : "M", "PreferBooks" : [ "frist book", "Second book" ] }
{ "_id" : ObjectId("567ac8e9498f43a9cea8dd09"), "Name" : "User2", "Age" : 2, "Gender" : "M", "PreferBooks" : [ "frist book", "Second book" ] }

索引如下:

> db.testcoll.getIndexes()
[
    {
        "v" : 1,
        "key" : {
            "_id" : 1
        },
        "name" : "_id_",
        "ns" : "testdb.testcoll"
    },
    {
        "v" : 1,
        "unique" : true,
        "key" : {
            "Name" : 1
        },
        "name" : "Name_1",
        "ns" : "testdb.testcoll"
    },
    {
        "v" : 1,
        "key" : {
            "Name" : 1,
            "Age" : 1
        },
        "name" : "Name_1_Age_1",
        "ns" : "testdb.testcoll"
    }
]
>

从索引里面看到有 Name:1,Age:1 的复合索引。查询单独 Age 数据:

> db.testcoll.find({Age:{$gt:80}}).hint("Name_1_Age_1").explain()
{
    "cursor" : "BtreeCursor Name_1_Age_1",
    "isMultiKey" : false,
    "n" : 21,
    "nscannedObjects" : 99,
    "nscanned" : 99,
    "nscannedObjectsAllPlans" : 99,
    "nscannedAllPlans" : 99,
    "scanAndOrder" : false,
    "indexOnly" : false,
    "nYields" : 0,
    "nChunkSkips" : 0,
    "millis" : 0,
    "indexBounds" : {
        "Name" : [
            [
                {
                    "$minElement" : 1
                },
                {
                    "$maxElement" : 1
                }
            ]
        ],
        "Age" : [
            [
                {
                    "$minElement" : 1
                },
                {
                    "$maxElement" : 1
                }
            ]
        ]
    },
    "server" : "junmao.local:27019",
    "filterSet" : false
}
>

上面结果显示的为什么 age 字段没有使用复合索引呢?麻烦请教下大家,复合索引创建在 Name:1 和 Age:1 的时候不是相当于 Name 和 Age 的都创建了么,查询 Age 数据最起码的应该是对 Age 进行索引啊,我查询 user 的时候是可以索引的,如下

> db.testcoll.find({Name:"User233"}).hint("Name_1_Age_1").explain()
{
    "cursor" : "BtreeCursor Name_1_Age_1",
    "isMultiKey" : false,
    "n" : 0,
    "nscannedObjects" : 0,
    "nscanned" : 0,
    "nscannedObjectsAllPlans" : 0,
    "nscannedAllPlans" : 0,
    "scanAndOrder" : false,
    "indexOnly" : false,
    "nYields" : 0,
    "nChunkSkips" : 0,
    "millis" : 0,
    "indexBounds" : {
        "Name" : [
            [
                "User233",
                "User233"
            ]
        ],
        "Age" : [
            [
                {
                    "$minElement" : 1
                },
                {
                    "$maxElement" : 1
                }
            ]
        ]
    },
    "server" : "junmao.local:27019",
    "filterSet" : false
}
>

同时查询 Name 和 Age :

> db.testcoll.find({Name:"User233",Age:233}).hint("Name_1_Age_1").explain()
{
    "cursor" : "BtreeCursor Name_1_Age_1",
    "isMultiKey" : false,
    "n" : 0,
    "nscannedObjects" : 0,
    "nscanned" : 0,
    "nscannedObjectsAllPlans" : 0,
    "nscannedAllPlans" : 0,
    "scanAndOrder" : false,
    "indexOnly" : false,
    "nYields" : 0,
    "nChunkSkips" : 0,
    "millis" : 0,
    "indexBounds" : {
        "Name" : [
            [
                "User233",
                "User233"
            ]
        ],
        "Age" : [
            [
                233,
                233
            ]
        ]
    },
    "server" : "junmao.local:27019",
    "filterSet" : false
}
>

复合索引也使用最左匹配法则了?

mongo version

MongoDB shell version: 2.6.11

真心请教

5976 次点击
所在节点    MongoDB
3 条回复
tracymcladdy
2015-12-26 19:29:10 +08:00
单独查询字段单独建索引
1yndonn3u
2015-12-27 22:45:11 +08:00
@tracymcladdy 复合索引不是相当于两条索引么?
1yndonn3u
2015-12-29 12:19:05 +08:00
@tracymcladdy 网上部分资料有问题,复合索引也是单独的索引。最左匹配原则,最左的相当于章,而第二个字段相当于节。

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/246290

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX