未使用Mongo索引 [英] Mongo Index not being used

查看:119
本文介绍了未使用Mongo索引的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我为正在执行的特定查询围绕多个项目创建了一个索引:

I created an index around several items for a particular query I am doing:

{
    "v" : 1,
    "key" : {
        "MODIFIED" : -1,
        "state" : 1,
        "fail" : 1,
        "generated" : 1
    },
    "ns" : "foo.bar",
    "name" : "MODIFIED_-1_state_1_fail_1_generated"
}

但是,当我执行查询时,似乎并没有使用我的索引.您能在我做错的事情上提供一些现场信息吗?

However when I execute my query, it doesn't apear to be using my index. Could you please provide some insite into what I'm doing wrong?

谢谢!

db.foo.find(    {
    "$or": [
        {
            "MODIFIED": {
                "$gt": {
                    "sec": 1321419600,
                    "usec": 0
                }
            }
        },
        {
            "$or": [
                {"state": "ca"},
                {"state": "ok"}
            ]
        }
    ],
    "$and": [
        {"fail": {"$ne": 1}},
        {"generated": {"$exists": false}}
    ]
}).explain();
{
    "cursor" : "BasicCursor",
    "nscanned" : 464215,
    "nscannedObjects" : 464215,
    "n" : 0,
    "millis" : 7549,
    "nYields" : 0,
    "nChunkSkips" : 0,
    "isMultiKey" : false,
    "indexOnly" : false,
    "indexBounds" : {

    }
}

推荐答案

您的索引不能用于您的查询是有充分的理由的,而且我还认为查询本身存在一些问题.之所以没有命中索引,是因为嵌套了$ or运算符,但我认为您的实际问题是对MongoDB中所有可用的运算符缺乏了解:

There's a good reason your index cannot be used for your query and I also think there are some issues with the query itself. The reason it's not hitting an index is because of the nested $or operator by the way but I think your actual problem is a lack of understanding on all the operators available to you in MongoDB :

首先,您不需要嵌套$或检查状态是"ca"还是"ok",(因为这是您未命中索引的主要原因)可以用state:{$in:["ca", "ok"]}代替确实做同样的事情.现在您的查询是:

First of all, your nested $or to check if the state is either "ca" or "ok" is not necessary and ( since it's the main reason you're not hitting your index) can be replaced with state:{$in:["ca", "ok"]} which does the exact same thing. Now your query is :

db.foo.find(    {
    "$or": [
        {
            "MODIFIED": {
                "$gt": {
                    "sec": 1321419600,
                    "usec": 0
                }
            }
        },
        {
            state:{$in:["ca", "ok"]}            
        }
    ],
    "$and": [
        {"fail": {"$ne": 1}},
        {"generated": {"$exists": false}}
    ]
}).explain();

它将击中您的索引.您的第二个问题是顶级$ and子句不是必需的.请注意AND(OR(A, B), AND(C, D)) = AND(OR(A, B), C, D).该查询执行相同的操作:

And it will hit your index. Your second issue is that a top-level $and clause is not necessary. Note that AND(OR(A, B), AND(C, D)) = AND(OR(A, B), C, D). This query does the same :

db.foo.find(    {
    "$or": [
        {
            "MODIFIED": {
                "$gt": {
                    "sec": 1321419600,
                    "usec": 0
                }
            }
        },
        {
            state:{$in:["ca", "ok"]}            
        }
    ],

    "fail": {"$ne": 1},
    "generated": {"$exists": false}

}).explain();

哪个仍然符合索引:

{
        "clauses" : [
                {
                        "cursor" : "BtreeCursor MODIFIED_-1_state_1_fail_1_generated_1 multi",
                        "nscanned" : 0,
                        "nscannedObjects" : 0,
                        "n" : 0,
                        "millis" : 1,
                        "nYields" : 0,
                        "nChunkSkips" : 0,
                        "isMultiKey" : false,
                        "indexOnly" : false,
                        "indexBounds" : {
                                "MODIFIED" : [
                                        [
                                                {
                                                        "$maxElement" : 1
                                                },
                                                {
                                                        "sec" : 1321419600,
                                                        "usec" : 0
                                                }
                                        ]
                                ],
                                "state" : [
                                        [
                                                {
                                                        "$minElement" : 1
                                                },
                                                {
                                                        "$maxElement" : 1
                                                }
                                        ]
                                ],
                                "fail" : [
                                        [
                                                {
                                                        "$minElement" : 1
                                                },
                                                1
                                        ],
                                        [
                                                1,
                                                {
                                                        "$maxElement" : 1
                                                }
                                        ]
                                ],
                                "generated" : [
                                        [
                                                null,
                                                null
                                        ]
                                ]
                        }
                },
                {
                        "cursor" : "BasicCursor",
                        "nscanned" : 0,
                        "nscannedObjects" : 0,
                        "n" : 0,
                        "millis" : 1,
                        "nYields" : 0,
                        "nChunkSkips" : 0,
                        "isMultiKey" : false,
                        "indexOnly" : false,
                        "indexBounds" : {

                        }
                }
        ],
        "nscanned" : 0,
        "nscannedObjects" : 0,
        "n" : 0,
        "millis" : 1
}

希望有帮助!顺便说一句,在复合索引中的第一个键以1的顺序开始,第二个键以-1的顺序开始更常规.请注意,-1仅用于确定相对于上一个字段的方向.

Hope that helps! By the way it's slightly more conventional to start the first key in your compound index with order 1 and the second with -1. Note that the -1 is only used to determine the direction relative to the previous field.

这篇关于未使用Mongo索引的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆