国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

java - 多線程并發(fā)情況下Map.containsKey() 判斷有問(wèn)題

瀏覽:132日期:2024-01-27 16:23:18

問(wèn)題描述

有下面一段代碼:

package test;import java.util.concurrent.ConcurrentHashMap;import java.util.concurrent.ConcurrentMap;public class TestContain extends Thread{ private final String key = 'key'; private final static ConcurrentMap<String, Object> locks = new ConcurrentHashMap<>();private static Object getLock(String lockName) { if (!locks.containsKey(lockName)) {//這一句會(huì)存在并發(fā)問(wèn)題locks.put(lockName, new String('我是值'));System.out.println('加了一次'); } return locks.get(lockName);}@Overridepublic void run() { getLock(this.key);};public static void main(String[] args) { for (int i = 0; i < 20; i++) {new TestContain().start();; }}}

輸出結(jié)果:

加了一次加了一次加了一次

表明了Map.containsKey() 在多線程的情況下會(huì)判斷不準(zhǔn)確。

這是為什么呢? 有什么方法改進(jìn)呢?

問(wèn)題解答

回答1:

ConcurrentHashMap的doc上有一段

Retrieval operations (including <tt>get</tt>) generally do not block, so may overlap with update operations (including

<tt>put</tt> and <tt>remove</tt>). Retrievals reflect the results of the most recently completed update operations holding upon their onset.

里面的get方法并不加鎖,get方法只是拿到最新完成update的值。

所以題主方法中的locks.containsKey(lockName)沒(méi)有鎖來(lái)保證線程安全的。而且感覺(jué)ConcurrentHashMap的使用場(chǎng)景并不是用containsKey來(lái)保證更新操作只進(jìn)行一次,而是用putIfAbsent來(lái)保證。

回答2:

ConcurrentMap保證的是單次操作的原子性,而不是多次操作。

你的getLock函數(shù)中包含了多次操作,ConcurrentMap沒(méi)法擴(kuò)大它的同步范圍,你需要自己實(shí)現(xiàn)getLock的鎖。

回答3:

使用putIfAbsent方法。

標(biāo)簽: java
相關(guān)文章:
主站蜘蛛池模板: 曲阜市| 陵川县| 化州市| 新绛县| 崇明县| 黑水县| 福建省| 开化县| 淮安市| 平泉县| 瓮安县| 永清县| 罗田县| 海门市| 静宁县| 那曲县| 皮山县| 闻喜县| 綦江县| 广宗县| 新密市| 察雅县| 辽中县| 普洱| 阿拉善右旗| 武宣县| 玉门市| 抚顺县| 安新县| 旬邑县| 大方县| 秭归县| 金平| 庄浪县| 绥滨县| 达孜县| 泸溪县| 怀安县| 肇源县| 公主岭市| 通海县|