MongoDB的索引与传统的关系型数据库几乎一模一样,极大提高了查询速度,MongoDB提供了多样性的索引支持,索引信息被保存在system.indexes中,MongoDB中_id字段在创建的时候,默认已建立了索引,这个索引比较特殊,并且不可删除,不过Capped Collections固定集合是个例外。

一、建立索引
  
test集合添加几条数据
    > db.test.insert({"userid":1,"name":"user1","age":20})
    > db.test.insert({"userid":2,"name":"user2","age":25})
    > db.test.insert({"userid":3,"name":"user3","age":26})
    .......
    1、创建普通索引
    建立索引的函数:ensureIndex() 数字1表示name键的索引按升序存储,-1表示索引按照降序方式存储。
    > db.test.ensureIndex({"name":1})
    当系统已有大量数据时,创建索引非常耗时,这时候可以在后台执行,需要指定background:true
    > db.test.ensureIndex({"name":1},{"background":true})
    可以通过下面的名称查看索引是否已经成功建立:
    > db.test.getIndexes()
    删除索引的命令是:
    > db.test.dropIndex({"name":1})
   2、唯一索引:
    在缺省情况下创建的索引均不是唯一索引。
    创建唯一索引
    > db.test.ensureIndex({"userid":1},{"unique":true})
    如果重复插入相同value的文档时,MongoDB将报错,以提示插入重复键,如:
    > db.test.insert({"userid":5,"name":"user5","age":20})
    > db.test.insert({"userid":5,"name":"user5","age":20})
    E11000 duplicate key error index: test.test.$userid_1  dup key: { : 5.0 }    
    如果插入的文档中不包含userid键,那么该文档中该键的值为null,如果多次插入类似的文档,MongoDB将会报出同样的错误,如:
    > db.test.insert({"userid1":5})
    > db.test.insert({"userid1":5})
    E11000 duplicate key error index: test.test.$userid_1  dup key: { : null }        
    如果在创建唯一索引时已经存在了重复项,我们可以通过下面的命令帮助我们在创建唯一索引时消除重复文档,仅保留发现的第一个文档,如:
    --先删除刚刚创建的唯一索引。
    > db.test.dropIndex({"userid":1})
    > db.test.insert({"userid":5,"name":"user5","age":20})
    > db.test.insert({"userid":5,"name":"user5","age":20})
    --创建唯一索引,并消除重复数据。
    > db.test.ensureIndex({"userid":1},{"unique":true,"dropDups":true})    
    --查询结果确认,重复的键确实在创建索引时已经被删除。
    > db.test.find()
    { "_id" : ObjectId("4fe823c180144abd15acd52e"), "userid" : 5 }
   
    3、复合索引
    在MongoDB中,我们同样可以创建复合索引,如:
    > db.test.ensureIndex({"name":1, "age":-1})
    该索引被创建后,基于name和age的查询将会用到该索引,或者是基于name的查询也会用到该索引,但是只是基于age的查询将不会用到该复合索引。因此可以说,如果想用到复合索引,必须在查询条件中包含复合索引中的前N个索引列。然而如果查询条件中的键值顺序和复合索引中的创建顺序不一致的话,MongoDB可以智能的帮助我们调整该顺序,以便使复合索引可以为查询所用。如:
    > db.test.find({"age": 30, "name": "stephen"})
    对于上面示例中的查询条件,MongoDB在检索之前将会动态的调整查询条件文档的顺序,以使该查询可以用到刚刚创建的复合索引。
    创建复合的唯一索引
    > db.test.ensureIndex({"userid":1,"age":1},{"unique":true})
    复合索引一定要注意索引的升降序

    4、为内嵌文档创建索引
    我们可以为内嵌文档创建索引,其规则和普通文档没有任何差别,如:
    > db.test.insert({"title":"aa",comments:{"date":"12","click":10}})
    > db.test.insert({"title":"bb",comments:{"date":"10","click":1}})   
    > db.test.ensureIndex({"comments.date":1})
    5、为排序创建索引
    随着集合的增长,需要针对查询中大量的排序做索引。如果没有对索引的键调用sort,MongoDB需要将所有数据提取到内存并排序。因此在做无索引排序时,如果数据量过大以致无法在内存中进行排序,此时MongoDB将会报错。
    6、索引名称
    对于上面创建的索引,MongoDB都会根据索引的keyname和索引方向为新创建的索引自动分配一个索引名,下面的命令可以在创建索引时为其指定索引名,如:
    > db.test.ensureIndex({"title":1},{"name":"testindex"})   

三、使用explain:
    explain是非常有用的工具,会帮助你获得查询方面诸多有用的信息。只要对游标调用该方法,就可以得到查询细节。explain会返回一个文档,而不是游标本身。如:
    > db.test.find().explain()
    {
        "cursor" : "BasicCursor",
        "nscanned" : 1,
        "nscannedObjects" : 1,
        "n" : 1,
        "millis" : 0,
        "nYields" : 0,
        "nChunkSkips" : 0,
        "isMultiKey" : false,
        "indexOnly" : false,
        "indexBounds" : {

        }    
    }
    explain会返回查询使用的索引情况,耗时和扫描文档数的统计信息。
    "cursor":"BasicCursor"表示没有使用索引。
    "nscanned":1 表示查询了多少个文档。
    "n":1 表示返回的文档数量。
    "millis":0 表示整个查询的耗时
二、索引管理
    system.indexes集合中包含了每个索引的详细信息,因此可以通过下面的命令查询已经存在的索引,如:
    > db.system.indexes.find()
   system.namespaces集合也含有所以的名字