我用 MongoDB 的 upsert 和$inc 实现了一个简单的自增计数器,但是却发现当对一个键并发自增计数时,会丢失数据,让人百思不得其解啊
我的 MongoDB 进程是在本机运行的,计数器主要代码如下: 计数器的 incr 实现:
public boolean incr(String key) {
checkKey(key);
UpdateResult result = mongoTemplate.upsert(query(Criteria.where("key").is(key)),
new Update().inc("value", 1),
MongoCounter.class, COLLECTION_NAME);
return result.wasAcknowledged();
}
测试代码:
int count = 200;
ExecutorService executor = Executors.newCachedThreadPool();
String key = "incrkey";
for (int i = 0; i < count; i++) {
executor.submit(() -> counter.incr(key));
}
assertTrue(counter.getCounter(key) == count);
executor.shutdown();
counter.delCounter(key);
在测试时,开了一个线程池,用多个线程同时向一个初始没有值的键连续调用 n 次 incr,但是结果发现总是会丢失几次更新,难道原子性更新的前提是并发量不能太大吗?
请懂 MongoDB 的人指点一二啊
这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。
V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。
V2EX is a community of developers, designers and creative people.