我试图了解在使用Node.js时如何确保在使用模型实例时的同步安全性。在这里,我在代码示例中使用Mongoose ODM,但问题适用于数据库与Node.js采用的异步事件驱动I / O方法一起使用的任何情况。
考虑以下代码(使用Mongoose进行MongoDB查询):
MyModel.findOne( { _id : <id #1> }, function( err, doc ) {
MyOtherModel.findOne( { _id : someOtherId }, ( function(err, otherDoc ) {
if (doc.field1 === otherDoc.otherField) {
doc.field2 = 0; // assign some new value to a field on the model
}
doc.save( function() { console.log( 'success' ); }
});
});
在应用程序的单独部分中,可以更新MyModel描述的文档。请考虑以下代码:
MyModel.update( { _id : <id #1> }, { $set : { field1 : someValue }, callback );
在Snippet A中,发出MongoDB查询,并在文档准备好后触发注册的回调。 MyModel描述的文档实例保留在内存中(在“doc”对象中)。可能会发生以下顺序:
虽然Node以单线程方式运行代码,但在我看来,任何允许运行的事件循环都会为可能过时的数据打开大门。如果这种遵守是错误的,请纠正我。
答案 0 :(得分:14)
不,不能保证node.js / MongoDB中不会出现这种竞争条件。它与node.js没有任何关系,这对任何支持并发访问的数据库都是可能的,而不仅仅是MongoDB。
然而,使用MongoDB解决问题比较棘手,因为它不支持像典型SQL数据库那样的事务。因此,您必须使用MongoDB cookbook here中概述的策略在应用程序层中解决它。