Mongodb并发与锁机制


1、 MongoDB 使用的锁


MongoDB 使用的是“readers-writer”锁, 可以支持并发但有很大的局限性,当一个读锁存在,许多

读操作可以使用这把锁,然而, 当一个写锁的存在,一个单一的写操作会 exclusively 持有该锁,同时

其它读,写操作不能使用共享这个锁;举个例子,假设一个集合里有 10 个文档,多个 update 操作不能

并发在这个集合上,即使是更新不同的文档。



2、 锁的粒度

在 2.2 版本以前,mongod 只有全局锁;在 2.2 版本开始,大部分读写操作只锁一个库,相对之前版本,

这个粒度已经下降,例如如果一个 mongod 实例上有 5 个库,如果只对一个库中的一个集合执行写操作,那

么在写操作过程中,这个库被锁;而其它 5 个库不影响。相比 RDBMS 来说,这个粒度已经算很大了!


3、 如何查看锁的状态

db.serverStatus()

db.currentOp()

mongotop

mongostat

the MongoDB Monitoring Service (MMS)


4、 哪些操作会对数据库产生锁?

下表列出了常见数据库操作产生的锁。

Changed in version 2.2.


Operation    Lock Type

Issue a query Read lock

Get more data from a cursor Read lock

Insert data Write lock

Remove data Write lock

Update data Write lock

Map-reduce    Read lock and write lock, unless operations are specified as non-atomic. Portions of map-reduce jobs can run concurrently.

Create an index Building an index in the foreground, which is the default, locks the database for extended periods of time.


db.eval()

Deprecated since version 3.0.

Write lock. The db.eval() method takes a global write lock while evaluating the JavaScript function. To avoid taking this global write lock, you can use the eval command with nolock: true.


eval

Deprecated since version 3.0.

Write lock. By default, eval command takes a global write lock while evaluating the JavaScript function. If used with nolock: true, the eval command does not take a global write lock while evaluating the JavaScript function. However, the logic within the JavaScript function may take write locks for write operations.

aggregate() Read lock


5、 哪些数据库管理操作会锁数据库?

某些数据库管理操作会 exclusively 锁住数据库,以下命令需要申请 exclusively 锁,并锁定一段时间


db.collection.ensureIndex(),

reIndex,

compact,

db.repairDatabase(),

db.createCollection(), when creating a very large (i.e. many gigabytes) capped collection,

db.collection.validate(),

db.copyDatabase().This operation may lock all databases 

 

以下命令需要申请 exclusively 锁,锁定数据库,但锁定很短时间。

db.collection.dropIndex(),

db.collection.getLastError(),

db.isMaster(),

rs.status() (i.e. replSetGetStatus,)

db.serverStatus(),

db.auth(), and

db.addUser().

 

备注:可见,一些查看命令也会锁库,在比较繁忙的生产库中,也会有影响的。



6、 锁住多个库的操作

以下数据库操作会锁定多个库。


db.copyDatabase() 锁定整个mongod实例

db.repairDatabase() 会获取全局写锁,运行期间会阻塞其它操作。

Journaling 内部操作,短时间锁定所有数据库,所有的数据库共享一个journal。

User authentication requires a read lock on the admin database for deployments using user credentials.

    For deployments authentication locks the admin database as well as the database the user is accessing.

All writes to a replica set’s primary lock both the database receiving the writes and then the local database for a short time. 

    The lock for the local database allows the mongod to write to the primary’s oplog and accounts for a small portion of the total time of the operation.


7、 JavaScript 操作的并发

Changed in version 2.4: The V8 JavaScript engine added in 2.4 allows multiple JavaScript operations to run at the same time. 

Prior to 2.4, a single mongod could only run a single JavaScript operation at once.

8、事务

Mongodb不支持多文档的事务


9、Mongodb的隔离保证

对于单一文档来说,读与写操作为原子操作,所以单一文档总是处于一致状态,这表示我们的读取操作,不会看到部分更新。

但是对于多文档来说,mongodb不会对并发写进行隔离:

因此,查询数据时,基于时间的数据快照是不能保证的。比如,我们获取不到查询过程当中已经删除的文档,或已经更新的文档。

Reads can see data which may subsequently be rolled back in rare cases such as replica set failover or power loss.

It does not mean that read operations can see documents in a partially written or otherwise inconsistent state.


Other systems refer to these semantics as read uncommitted.

参考:http://liyanblog.cn/articles/2013/01/15/1358218545015.html

参考:http://docs.mongodb.org/manual/faq/concurrency/