不安全的HashMap

不安全的HashMap

在多线程使用场景中,应该尽量避免使用线程不安全的HashMap,而使用线程安全的ConcurrentHashMap。那么为什么说HashMap是 线程不安全的,下面举例子说明在并发的多线程使用场景中使用HashMap可能造成死循环。

 public class HashMapInfiniteLoop {

    private static HashMap<Integer,String> map = new HashMap<Integer,String>(20.75f);
    public static void main(String[] args) {
        map.put(5"C");

        new Thread("Thread1") {
            public void run() {
                map.put(7, "B");
                System.out.println(map);
            };
        }.start();

        new Thread("Thread2") {
            public void run() {
                map.put(3, "A);
                System.out.println(map);
            };
        }.start();
    }
}

其中,map初始化为一个长度为2的数组,loadFactor=0.75,threshold=2*0.75=1,也就是说当put第二个key的时候,map就需要进行resize。通过设置断点让线程1和线程2同时debugtransfer方法的首行。注意此时两个线程已经成功添加数据。放开thread1的断点至transfer方法的“Entry next = e.next;” 这一行;然后放开线程2的的断点,让线程2进行resize。结果如下图。

断点调试图

注意,Thread1e指向了key(3),而next指向了key(7),其在线程二rehash后,指向了线程二重组后的链表。线程一被调度回来执行,先是执行newTalbe[i] = e, 然后是e = next,导致了e指向了key(7),而下一次循环的next = e.next导致了next指向了key(3)

断点调试图

e.next = newTable[i]导致key(3).next指向了key(7)。注意:此时的key(7).next已经指向了key(3), 环形链表就这样出现了。

循环示意图

于是,当我们用线程一调用map.get(11)时,悲剧就出现了:Infinite Loop。

下一页