3

我了解复合多键索引可能仅包含一个数组字段

以下不会产生“无法索引并行数组”错误:

db.test.ensureIndex({"values.x": 1, "values.y": 1})

db.test.insert({"values": [ {"x": 1, "y": 2}, {"x": 2, "y": 2} ]})
db.test.insert({"values": [ {"x": 2, "y": 1}, {"x": 1, "y": 1} ]})
db.test.insert({"values": [ {"x": 1, "y": 1}, {"x": 1, "y": 1} ]})

因此,在对象嵌套在一个数组字段中的多个对象属性上,似乎允许使用复合索引。

文档说“MongoDB 分别索引数组中的每个值”,因此对于上述场景,我希望为每个文档中的 values.x 和 values.y 的所有组合创建索引条目。

但是,以下对两个嵌套字段的查询表明仅使用复合索引中的第一个字段 - nscanned 为 2 表明 Mongo 必须检查第二个添加的文档以检查与 x = 匹配的数组元素上的 y = 2 2.

db.test.find({"values.x": 2, "values.y": 2}).explain()
{
    "cursor" : "BtreeCursor values.x_1_values.y_1",
    "isMultiKey" : true,
    "n" : 1,
    "nscannedObjects" : 2,
    "nscanned" : 2,
    "nscannedObjectsAllPlans" : 2,
    "nscannedAllPlans" : 2,
    "scanAndOrder" : false,
    "indexOnly" : false,
    "nYields" : 0,
    "nChunkSkips" : 0,
    "millis" : 0,
    "indexBounds" : {
            "values.x" : [
                    [
                            2,
                            2
                    ]
            ],
            "values.y" : [
                    [
                            {
                                    "$minElement" : 1
                            },
                            {
                                    "$maxElement" : 1
                            }
                    ]
            ]
    },
    "server" : "localhost:27017"
}

MongoDB 索引和复合索引对仅覆盖第一个字段的值有什么价值?

4

1 回答 1

2

如果您使用$elemMatch查询运算符在同一元素中搜索xy值,您会看到索引边界也适用于y

> db.test.find({ values: { $elemMatch: { x: 2, y: 2 }}}).explain()
{
    "cursor" : "BtreeCursor values.x_1_values.y_1",
    "isMultiKey" : true,
    "n" : 1,
    "nscannedObjects" : 1,
    "nscanned" : 1,
    "nscannedObjectsAllPlans" : 1,
    "nscannedAllPlans" : 1,
    "scanAndOrder" : false,
    "indexOnly" : false,
    "nYields" : 0,
    "nChunkSkips" : 0,
    "millis" : 0,
    "indexBounds" : {
        "values.x" : [
            [
                2,
                2
            ]
        ],
        "values.y" : [
            [
                2,
                2
            ]
        ]
    },
    "server" : "localhost:27017"
}

这是在SERVER-3104中为 2.4 实现的。票证说明解释了为什么这些索引范围不能用于您的原始查询:

在多个字段上创建复合索引时,Mongo 不会计算笛卡尔积。如果根据 index 对文档{ a:[ { b:1, c:2 }, { b:10, c:20 } ] }进行索引{ 'a.b':1, 'a.c':1 },则创建的索引键是{ '':1, '':2 }{ '':10, '':20 }。(例如,没有索引键 { '':1, '':20 }。)

现在,假设我们有一个查询{ 'a.b':1, 'a.c':20 }。该查询应该与文档匹配,因为文档中存在 'ab' 值为 1,而文档中存在 'ac' 值为 20。但是,没有索引键在“ab”位置包含 1,在“ac”位置包含 20。因此,'ab' 上的索引范围将是[[ 1, 1 ]],但 'ac' 上不会有任何索引范围。这意味着{ '':1, '':2 }将检索索引键并用于查找完整文档,并且匹配器将确定完整文档与查询匹配

于 2013-09-05T17:50:36.877 回答