zoukankan      html  css  js  c++  java
  • ConcurrentHashMap并不是绝对线程安全的

    import java.util.HashMap;
    import java.util.Map;
    import java.util.concurrent.ConcurrentHashMap;
    
    public class ConcurrentHashMapDemo1 {
    
        private static Map<Long, Integer> widgetCacheMap = new ConcurrentHashMap<Long, Integer>();
    
        /**
         * @param args
         */
        public static void main(String[] args) {
            for (int i = 0; i < 10000; i++) {
                Thread tt = new Thread(new Rund());
                tt.start();
            }
        }
    
        static class Rund implements Runnable {
    
            public void run() {
                test();
            }
    
            /**
             * 1W次,总有那么几次线程不安全
             */
            public void test() {
                synchronized ("") {// 解决方案
                    ConcurrentHashMapDemo1 tt = new ConcurrentHashMapDemo1();
                    tt.set();
                    int s1 = widgetCacheMap.get(1L).intValue();
                    tt.change();
                    int s2 = widgetCacheMap.get(1L).intValue();
                    if (s1 == s2) {
                        System.out.println(s1 + ":" + s2);
                    }
                }
            }
    
        }
    
        public void set() {
            Map<Long, Integer> mm = new HashMap<Long, Integer>();
            mm.put(1L, 1);
            widgetCacheMap = mm;
        }
    
        public void change() {
            Map<Long, Integer> mm = new HashMap<Long, Integer>();
            mm.put(1L, 2);
            widgetCacheMap = mm;
        }
    
    }
    import java.util.concurrent.ConcurrentHashMap;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    import java.util.concurrent.TimeUnit;
    
    public class ConcurrentHashMapDemo2 {
    
        public static void main(String[] args) throws InterruptedException {
            for (int i = 0; i < 10; i++) {
                System.out.println(test());
            }
        }
    
        private static int test() throws InterruptedException {
            ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<String, Integer>();
            ExecutorService pool = Executors.newCachedThreadPool();
            for (int i = 0; i < 8; i++) {
                pool.execute(new MyTask(map));
            }
            pool.shutdown();
            pool.awaitTermination(1, TimeUnit.DAYS);
    
            return map.get(MyTask.KEY);
        }
    }
    
    class MyTask implements Runnable {
    
        public static Object lock = new Object();
    
        public static final String KEY = "key";
    
        private ConcurrentHashMap<String, Integer> map;
    
        public MyTask(ConcurrentHashMap<String, Integer> map) {
            this.map = map;
        }
    
        @Override
        public void run() {
            for (int i = 0; i < 100; i++) {
                synchronized (lock) { // 解决方案
                    this.addup();
                }
            }
        }
    
        private void addup() {
            if (!map.containsKey(KEY)) {
                map.put(KEY, 1);
            } else {
                map.put(KEY, map.get(KEY) + 1);
            }
        }
    }

    总结:ConcurrentHashMap是线程安全的,那是在他们的内部操作,其外部操作还是需要自己来保证其同步的,特别是静态的ConcurrentHashMap,其有更新和查询的过程,要保证其线程安全,需要syn一个不可变的参数才能保证其原子性

  • 相关阅读:
    初窥RabbitMQ消息中间及SpringBoot整合
    博客园页面嵌入左下角小女孩
    不依赖Spring使用AspectJ达到AOP面向切面编程
    彻底搞懂计算机网络通信设备与协议
    博客园如何嵌入网易云音乐播放器
    并发编程之详解InheritableThreadLocal类原理
    2小时学会Spring Boot(IDE:eclipse)
    $.Ajax、$.Get、$.Post代码实例参数解析
    设计模式之空对象模式
    设计模式之黑板模式
  • 原文地址:https://www.cnblogs.com/sprinng/p/6627713.html
Copyright © 2011-2022 走看看