导致插入文档的ups不是完全原子的操作。将ups视为执行以下离散步骤:
- 查询标识的文档以进行增补。
- 如果文档存在,则自动更新现有文档。
- 否则(该文档不存在),自动插入一个包含查询字段和更新的新文档。
因此,第2步和第3步都是原子操作,但是在第1步之后可能会发生另一个upsert,因此您的代码需要检查重复的键错误,然后再次尝试upsert。到那时,您知道
_id存在的文档,因此它将始终成功。
例如:
var minute = utils.minute();Monitor.update({ _id: minute }, { $inc: update }, { upsert: true }, function(err) { if (err) { if (err.pre === 11000) { // Another upsert occurred during the upsert, try again. You could omit the // upsert option here if you don't ever delete docs while this is running. Monitor.update({ _id: minute }, { $inc: update }, { upsert: true }, function(err) { if (err) { console.trace(err); } }); } else { console.trace(err); } }});有关相关文档,请参见此处。
您可能仍然想知道,如果插入是原子的,为什么会发生这种情况,但这意味着插入的文档在完全写入之前不会进行任何更新,而不是不会
_id出现具有相同文档的其他插入。
另外,您不需要手动创建索引,
_id因为所有MongoDB集合都具有唯一的索引
_id。因此,您可以删除以下行:
monitorSchema.index({_id: -1}); // Not needed


