gpt4 book ai didi

Java ConcurrentHashMap 操作原子性

转载 作者:搜寻专家 更新时间:2023-10-30 21:05:46 24 4
gpt4 key购买 nike

这可能是一个重复的问题,但我在一本关于并发的书中找到了这部分代码。这据说是线程安全的:

ConcurrentHashMap<String, Integer> counts = new ...;

private void countThing(String thing) {
while (true) {
Integer currentCount = counts.get(thing);
if (currentCount == null) {
if (counts.putIfAbsent(thing, 1) == null)
break;
} else if (counts.replace(thing, currentCount, currentCount + 1)) {
break;
}
}
}

从我(并发初学者)的角度来看,线程 t1 和线程 t2 都可以读取 currentCount = 1。然后两个线程都可以将映射的值更改为 2。有人可以解释一下代码是否正确吗?

最佳答案

诀窍在于 replace(K key, V oldValue, V newValue) 为您提供了原子性。来自 the docs (强调我的):

Replaces the entry for a key only if currently mapped to a given value. ... the action is performed atomically.

关键词是“原子地”。在 replace 中,“检查旧值是否符合我们的预期,如果是,则替换它”作为一个单独的工作 block 发生,没有其他线程能够与之交错。执行需要执行任何同步以确保它提供这种原子性。

因此,不可能两个线程都从 replace 函数中看到 currentAction == 1。其中之一会将其视为 1,因此对 replace 的调用将返回 true。另一个会将其视为 2(因为第一次调用),因此返回 false — 并循环返回重试,这次使用 currentAction == 2 的新值。

当然,这可能是第三个线程同时将 currentAction 更新为 3,在这种情况下,第二个线程将继续尝试,直到幸运地没有任何人跳到它前面为止。

关于Java ConcurrentHashMap 操作原子性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38165869/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com