mongo索引

Indexes

Working with indexes in the MongoDB ODM is pretty straight forward. You can have multiple indexes, they can consist of multiple fields, they can be unique and you can give them an order. In this chapter we’ll show you examples of indexes using annotations.

First here is an example where we put an index on a single property:

  • PHP
    <?php
    
    namespace Documents;
    
    /** @Document */
    class User
    {
        /** @Id */
        public $id;
    
        /** @Field(type="string") @Index */
        public $username;
    }
    
  • XML
  • YAML

Index Options

You can customize the index with some additional options:

  • name - The name of the index. This can be useful if you are indexing many keys and Mongo complains about the index name being too long.
  • dropDups - If a unique index is being created and duplicate values exist, drop all but one duplicate value.
  • background - Create indexes in the background while other operations are taking place. By default, index creation happens synchronously. If you specify TRUE with this option, index creation will be asynchronous.
  • safe - You can specify a boolean value for checking if the index creation succeeded. The driver will throw a MongoCursorException if index creation failed.
  • expireAfterSeconds - If you specify this option then the associated document will be automatically removed when the provided time (in seconds) has passed. This option is bound to a number of limitations, which are documented at https://docs.mongodb.com/manual/tutorial/expire-data/.
  • order - The order of the index (asc or desc).
  • unique - Create a unique index.
  • sparse - Create a sparse index. If a unique index is being created the sparse option will allow duplicate null entries, but the field must be unique otherwise.
  • partialFilterExpression - Create a partial index. Partial indexes only index the documents in a collection that meet a specified filter expression. By indexing a subset of the documents in a collection, partial indexes have lower storage requirements and reduced performance costs for index creation and maintenance. This feature was introduced with MongoDB 3.2 and is not available on older versions.

Unique Index(唯一索引)

  • PHP
    <?php
    
    namespace Documents;
    
    /** @Document */
    class User
    {
        /** @Id */
        public $id;
    
        /** @Field(type="string") @Index(unique=true, order="asc") */
        public $username;
    }
    
  • XML
  • YAML

For your convenience you can quickly specify a unique index with @UniqueIndex:

  • PHP
    <?php
    
    namespace Documents;
    
    /** @Document */
    class User
    {
        /** @Id */
        public $id;
    
        /** @Field(type="string") @UniqueIndex(order="asc") */
        public $username;
    }
    
  • XML
  • YAML

If you want to specify an index that consists of multiple fields you can specify them on the class doc block:

  • PHP
    <?php
    
    namespace Documents;
    
    /**
     * @Document
     * @UniqueIndex(keys={"accountId"="asc", "username"="asc"})
     */
    class User
    {
        /** @Id */
        public $id;
    
        /** @Field(type="int") */
        public $accountId;
    
        /** @Field(type="string") */
        public $username;
    }
    
  • XML
  • YAML

To specify multiple indexes you must use the @Indexes annotation:

  • PHP
    <?php
    
    /**
     * @Document
     * @Indexes({
     *   @Index(keys={"accountId"="asc"}),
     *   @Index(keys={"username"="asc"})
     * })
     */
    class User
    {
        /** @Id */
        public $id;
    
        /** @Field(type="int") */
        public $accountId;
    
        /** @Field(type="string") */
        public $username;
    }
    
  • XML
  • YAML

Embedded Indexes

You can specify indexes on embedded documents just like you do on normal documents. When Doctrine creates the indexes for a document it will also create all the indexes from its mapped embedded documents.

<?php

namespace Documents;

/** @EmbeddedDocument */
class Comment
{
    /** @Field(type="date") @Index */
    private $date;

    // ...
}

Now if we had a BlogPost document with the Comment document embedded many times:

<?php

namespace Documents;

/** @Document */
class BlogPost
{
    // ...

    /** @Field(type="string") @Index */
    private $slug;

    /** @EmbedMany(targetDocument="Comment") */
    private $comments;
}

If we were to create the indexes with the SchemaManager:

<?php

$sm->ensureIndexes();

It will create the indexes from the BlogPost document but will also create the indexes that are defined on the Comment embedded document. The following would be executed on the underlying MongoDB database:

db.BlogPost.ensureIndexes({ ‘slug’ : 1, ‘comments.date’: 1 })

Also, for your convenience you can create the indexes for your mapped documents from the console:

$ php mongodb.php mongodb:schema:create –index

If you are mixing document types for your embedded documents, ODM will not be able to create indexes for their fields unless you specify a discriminator map for the embed-one or embed-many relationship.

Geospatial Indexing

You can specify a geospatial index by just specifying the keys and options structures manually:

  • PHP
    <?php
    
    /**
     * @Document
     * @Index(keys={"coordinates"="2d"})
     */
    class Place
    {
        /** @Id */
        public $id;
    
        /** @EmbedOne(targetDocument="Coordinates") */
        public $coordinates;
    }
    
    /** @EmbeddedDocument */
    class Coordinates
    {
        /** @Field(type="float") */
        public $latitude;
    
        /** @Field(type="float") */
        public $longitude;
    }
    
  • XML
  • YAML

Partial indexes

You can create a partial index by adding a partialFilterExpression to any index.

  • PHP
    <?php
    
    /**
     * @Document
     * @Index(keys={"city"="asc"}, partialFilterExpression={"version"={"$gt"=1}})
     */
    class Place
    {
        /** @Id */
        public $id;
    
        /** @Field(type="string") */
        public $city;
    
        /** @Field(type="int") */
        public $version;
    }
    
  • XML
  • YAML

Partial indexes are only available with MongoDB 3.2 or newer. For more information on partial filter expressions, read the official MongoDB documentation.

Requiring Indexes

Requiring Indexes was deprecated in 1.2 and will be removed in 2.0.

Sometimes you may want to require indexes for all your queries to ensure you don’t let stray unindexed queries make it to the database and cause performance problems.

  • PHP
    <?php
    
    /**
     * @Document(requireIndexes=true)
     */
    class Place
    {
        /** @Id */
        public $id;
    
        /** @Field(type="string") @Index */
        public $city;
    }
    
  • XML
  • YAML

When you run queries it will check that it is indexed and throw an exception if it is not indexed:

<?php

$qb = $dm->createQueryBuilder('Documents\Place')
    ->field('city')->equals('Nashville');
$query = $qb->getQuery();
$places = $query->execute();

When you execute the query it will throw an exception if city was not indexed in the database. You can control whether or not an exception will be thrown by using the requireIndexes() method:

<?php

$qb->requireIndexes(false);

You can also check if the query is indexed and with the isIndexed() method and use it to display your own notification when a query is unindexed:

<?php

$query = $qb->getQuery();
if (!$query->isIndexed()) {
    $notifier->addError('Cannot execute queries that are not indexed.');
}

If you don’t want to require indexes for all queries you can set leave requireIndexes as false and control it on a per query basis:

<?php

$qb->requireIndexes(true);
$query = $qb->getQuery();
$results = $query->execute();
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值