av一区二区在线观看_亚洲男人的天堂网站_日韩亚洲视频_在线成人免费_欧美日韩精品免费观看视频_久草视

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

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

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

問題描述

有下面一段代碼:

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)) {//這一句會存在并發(fā)問題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() 在多線程的情況下會判斷不準(zhǔn)確。

這是為什么呢? 有什么方法改進(jì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)沒有鎖來保證線程安全的。而且感覺ConcurrentHashMap的使用場景并不是用containsKey來保證更新操作只進(jìn)行一次,而是用putIfAbsent來保證。

回答2:

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

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

回答3:

使用putIfAbsent方法。

標(biāo)簽: java
相關(guān)文章:
主站蜘蛛池模板: 三级视频在线观看电影 | 亚洲视频在线看 | 青青久久av北条麻妃海外网 | 亚洲成人精品久久久 | 欧美精品一区二区三区视频 | 91视视频在线观看入口直接观看 | 日日操日日舔 | 亚洲经典一区 | 日韩精品一区二区三区中文在线 | 亚洲综合无码一区二区 | 亚洲国产欧美在线人成 | 日韩欧美一区二区三区四区 | 欧美综合一区 | 最近中文字幕免费 | 国产乱码精品一区二区三区忘忧草 | 久久精品视频网站 | av片在线观看网站 | 天天操天天玩 | 日本特黄特色aaa大片免费 | 第一区在线观看免费国语入口 | 久操福利 | 丝袜 亚洲 另类 欧美 综合 | 天天碰日日操 | 久久久久国产一区二区三区不卡 | 精品国产1区2区3区 在线国产视频 | 97超碰成人| h视频免费在线观看 | 一区二区高清 | www.日韩| 国产视频久 | 成人在线一区二区三区 | 91在线导航 | 亚洲成人一二区 | 国产女人与拘做受免费视频 | 五月婷婷丁香婷婷 | 无码日韩精品一区二区免费 | 久久性| 久久国产精品免费视频 | 一区二区三区精品在线视频 | 黄免费观看| 久久国色|