- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
Java 8 中引入了一个新的 computeIfAbsent API。ConcurrentHashMap's impelementation of it 的 javadocs状态:
If the specified key is not already associated with a value, attempts to compute its value using the given mapping function and enters it into this map unless null. The entire method invocation is performed atomically, so the function is applied at most once per key. Some attempted update operations on this map by other threads may be blocked while computation is in progress, so the computation should be short and simple, and must not attempt to update any other mappings of this map.
那么,在 key 已经存在且不需要计算的情况下,锁定此实现有何意义?整个方法 computeIfAbsent 是否如文档中所述同步,即使不需要计算或仅同步映射函数调用以防止调用函数两次?
最佳答案
执行ConcurrentHashMap非常复杂,因为它专门设计用于允许并发可读性,同时最大限度地减少更新争用。在一个非常高的抽象层次上,它被组织为一个分桶哈希表。所有读取操作都不需要锁定,并且(引用 javadoc)“不支持以阻止所有访问的方式锁定整个表”。为了实现这一点,内部设计非常复杂(但仍然优雅),键值映射保存在节点中,节点可以以各种方式(例如列表或平衡树)排列,以利用细粒度锁。如果您对实现细节感兴趣,您还可以查看 source code .
尝试回答您的问题:
So, what does it say about locking of this implementation in case when the the key already exists and the computation is unneeded?
可以合理地认为,与任何读取操作一样,不需要锁定来检查键是否已经存在并且不需要执行映射函数。
Is the whole method computeIfAbsent synchronized as stated in docs even if no calculation is needed or just the mapping function call is synchronized to prevent calling the function twice?
不,该方法在锁定方面不是同步,但从调用者的角度来看,它是原子执行的(即映射函数最多应用一次)。如果未找到键,则必须使用映射函数计算的值执行更新操作,并且在调用该函数时涉及某种锁定。可以合理地认为这种锁定是非常细粒度的并且只涉及表的很小一部分(嗯,必须存储 key 的特定数据结构)这就是原因(引用 javadoc,强调我的) “一些 其他线程尝试的更新操作可能在计算过程中被阻止”。
关于java - ConcurrentHashMap computeIfAbsent,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26481796/
我有一个 Hashmap,在其中维护不同类型的读取器到它们各自的 java 类实现的映射。我有一个多线程 Java 服务器,支持 32 种类型的阅读器。 您可以假设每 30 秒,每种类型的读取器都会有
Java 8 中引入了一个新的 computeIfAbsent API。ConcurrentHashMap's impelementation of it 的 javadocs状态: If the s
在本文中,我们将学习 Java ConcurrentHashMap 类的 computeIfAbsent() 方法。找到 computeIfAbsent 方法的 Java 文档。 V computeI
computeIfAbsent 是 java.util.Map 的默认方法,已在 Java 8 中引入。当与指定键关联的值不可用或为 null 时,computeIfAbsent 方法起作用,在这种情
我有这样的代码: ConcurrentMap map = new ConcurrentHashMap<>(); AtomicBoolean isNew = new AtomicBoolean(fals
java.util.HashMap 的文档明确指出:“如果多个线程同时访问一个 HashMap ,并且至少有一个线程在结构上修改了该映射,则它必须在外部进行同步。” 但是,请考虑使用映射作为缓存以减少
升级到 Android Studio 3 Canary 4 后出现此错误: Gradle sync failed: Argument for @NotNull parameter 'key' of c
考虑以下某种固定大小缓存的实现,它允许通过整数句柄进行查找: static class HandleCache { private final AtomicInteger counter =
我是 Java 的新手,有点从 C# 过渡到 Java。 java.util.function有一个定义为 Function 的接口(interface)这是输入到 computeIfAbsent M
于是我对 Java 的 Map computeIfAbsent 产生了好奇。 (使用 java8)方法,我希望有人能告诉我为什么会这样,因为我无法真正理解该问题背后的逻辑。 所以,我有一个带有键的映射
我很想用Map.computeIfAbsent但是距离本科生的lambda已经太久了。 几乎直接来自文档:它提供了一个旧方法的示例: Map whoLetDogsOut = new Concurren
我正在尝试制作具有以下功能的工厂: 它应该总是返回一个 Filter 对象。 从 hashmap = 如果字符串字母(键)已经存在于 hashmap 中,它应该只从 hashmap 中获取它的值(对象
我有以下代码,它是一个玩具代码,但可以重现该问题: import java.util.*; import java.util.concurrent.ConcurrentHashMap; import
ConcurrentHashMap#computeIfAbsent 中的 Javadoc 说 The computation should be short and simple, and must
整个方法调用是原子的还是只是 BiFunction 执行的原子?是阻塞所有键还是只阻塞同一键上的调用? 最佳答案 以下详细信息适用于 OpenJDK Java 11。 这三种方法持有 Node 的锁在
我正在练习 Cay S. Horstmann 所著的“Java SE 8 for the really Impatient”一书的练习。有 2 个练习要求相同算法的不同实现,一个使用 merge ,其
我正在使用 ConcurrentHashMap在 Scala 中,我想使用 computeIfAbsent()方法,但无法弄清楚第二个参数的语法。有人可以告诉我什么是正确的语法吗? 运行以下代码时 v
有没有办法模拟ConcurrentHashmap.computeIfAbsent但不将条目分配给hashmap。我只需要当 HashMap 中的条目尚不存在时该方法生成的实例。这些都在线程中运行。所以
我有一组非常适合我的代码: for (String word : distinctWordsInOneLigne) { Map> map = new H
大家好,我想知道如何在不使用 lambda 表达式的情况下使用 JSONObject.computeIfAbsent 。实际上,这是我的代码 list.computeIfAbsent(id, k ->
我是一名优秀的程序员,十分优秀!