java - 多線程并發情況下Map.containsKey() 判斷有問題
問題描述
有下面一段代碼:
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)) {//這一句會存在并發問題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();; }}}
輸出結果:
加了一次加了一次加了一次
表明了Map.containsKey() 在多線程的情況下會判斷不準確。
這是為什么呢? 有什么方法改進呢?
問題解答
回答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來保證更新操作只進行一次,而是用putIfAbsent來保證。
回答2:ConcurrentMap保證的是單次操作的原子性,而不是多次操作。
你的getLock函數中包含了多次操作,ConcurrentMap沒法擴大它的同步范圍,你需要自己實現getLock的鎖。
回答3:使用putIfAbsent方法。
相關文章:
1. 一個mysql聯表查詢的問題2. 運行python程序時出現“應用程序發生異常”的內存錯誤?3. html5 - iOS的webview加載出來的H5網頁,怎么修改html標簽select的樣式字體?4. java - 我現在一個servlet中有調用socket訪問,作為socket的客戶端,5. python 如何實現PHP替換圖片 鏈接6. python如何不改動文件的情況下修改文件的 修改日期7. 主從備份 - 跪求mysql 高可用主從方案8. javascript - git clone 下來的項目 想在本地運行 npm run install 報錯9. mysql主從 - 請教下mysql 主動-被動模式的雙主配置 和 主從配置在應用上有什么區別?10. mysql - 一個表和多個表是多對多的關系,該怎么設計
