我参考的是这篇文章: https://www.cnkirito.moe/java-ConcurrentHashMap-CAS/ 网上论述这个的文章实在是不多。
我的测试代码如下:
public class test2 { private final static Map<String, Table> map = new ConcurrentHashMap<>(); private static final String KEY = "key";
public static void increase1(String key) {
Table oldTable = map.get(key);
int value = oldTable.getI();
oldTable.setI(value + 1);
map.put(key, oldTable);
}
public static void increase2(String key) {
Table oldTable;
Table newTable = new Table(0);
while(true) {
oldTable = map.get(KEY);
newTable.setI(oldTable.getI() + 1);
if(map.replace(KEY, oldTable, newTable)) break;
}
}
public static int getTableValue(String key) {
return map.get(key).getI();
}
public static void main(String[] args) {
map.put(KEY, new Table(0));
ExecutorService executor = Executors.newFixedThreadPool(10);
int callTime = 1000;
CountDownLatch countDownLatch = new CountDownLatch(callTime);
for(int i=0; i<callTime; i++) {
executor.execute(new Runnable() {
@Override
public void run() {
increase2(KEY);
countDownLatch.countDown();
}
});
}
try {
countDownLatch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
executor.shutdown();
System.out.println("调用次数:" + getTableValue(KEY));
}
}
class Table { private int i;
public Table(int i) {
this.i = i;
}
public int getI() {
return i;
}
public void setI(int i) {
this.i = i;
}
}
测试结果:使用 increase1 的话,调用次数是不停变动的,存在并发错误。而用 increase2 的话,恒定都是 1000.
这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。
V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。
V2EX is a community of developers, designers and creative people.