一尘不染

迭代ConcurrentHashMap值线程安全吗?

java

在javadoc中,ConcurrentHashMap如下:

检索操作(包括get)通常不会阻塞,因此可能与更新操作(包括put和remove)重叠。检索反映了自启动以来最新完成的更新操作的结果。对于诸如putAll和clear的聚合操作,并发检索可能仅反映某些条目的插入或删除。同样,迭代器和枚举返回的元素反映了在创建迭代器/枚举时或此后某个时刻哈希表的状态。他们不抛出ConcurrentModificationException。但是,迭代器被设计为一次只能由一个线程使用。

这是什么意思?如果我尝试同时使用两个线程迭代地图,会发生什么情况?如果在迭代过程中从地图上放置或删除值,会发生什么情况?


阅读 354

收藏
2020-03-21

共1个答案

一尘不染

这是什么意思?

这意味着你从中获得的每个迭代器ConcurrentHashMap都旨在由单个线程使用,并且不应被传递。这包括for-each循环提供的语法糖。

如果我尝试同时使用两个线程迭代地图,会发生什么情况?

如果每个线程都使用自己的迭代器,它将按预期工作。

如果在迭代过程中从地图上放置或删除值,会发生什么情况?

保证这样做不会使事情中断(这是“并发” ConcurrentHashMap含义的一部分)。但是,不能保证一个线程将看到另一线程执行的映射更改(无需从映射中获取新的迭代器)。保证迭代器在创建地图时能够反映其状态。进一步的更改可能会反映在迭代器中,但不一定如此。

总之,类似

for (Object o : someConcurrentHashMap.entrySet()) {
// …
}
几乎每次看到它都会好(或至少很安全)。

2020-03-21