java - 多線程并發(fā)情況下Map.containsKey() 判斷有問(wèn)題
問(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方法。
相關(guān)文章:
1. android - NavigationView 的側(cè)滑菜單中如何保存新增項(xiàng)(通過(guò)程序添加)2. mysql - select查詢(xún)多個(gè)紀(jì)錄的條件怎么寫(xiě)3. linux - 編譯安裝mysql 5.6.234. 提示語(yǔ)法錯(cuò)誤語(yǔ)法錯(cuò)誤: unexpected ’abstract’ (T_ABSTRACT)5. 這段代碼既不提示錯(cuò)誤也看不到結(jié)果,請(qǐng)老師明示錯(cuò)在哪里,謝謝!6. php7.3.4中怎么開(kāi)啟pdo驅(qū)動(dòng)7. 老師 我是一個(gè)沒(méi)有學(xué)過(guò)php語(yǔ)言的準(zhǔn)畢業(yè)生 我希望您能幫我一下8. ueditor上傳服務(wù)器提示后端配置項(xiàng)沒(méi)有正常加載,求助!!!!!9. tp5 不同控制器中的變量調(diào)用問(wèn)題10. php - 第三方支付平臺(tái)在很短時(shí)間內(nèi)多次異步通知,訂單多次確認(rèn)收款
