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一个不可变的参数才能保证其原子性

posted @ 2017-03-27 17:17  Earic  阅读(6409)  评论(0编辑  收藏  举报