ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

JDK8 HashMap也会死循环

2021-10-01 22:30:16  阅读:193  来源: 互联网

标签:util java HashMap 11.0 链表 JDK8 base 死循环


前言

  • 我们知道JDK 7 HashMap是数组+链表数据结构,并发插入条件下,因为链表采用头插法,可能造成循环链表的问题,导致cpu占用100%
  • JDK8 HashMap是数组+链表+红黑树的数据结构,应为链表改用尾插法,所有并发插入时没有了循环链表问题。但是红黑树一样会有死循环问题
  • 注意:HashMap并发条件下不止有死循环问题,还可能丢数据等等

问题复现

public class Test {

    private static Map<String, String> map = new HashMap<>();

    @SneakyThrows
    public static void main(String[] args) {
        CountDownLatch latch = new CountDownLatch(2);
        // 线程1 => t1
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 100_0000; i++) {
                    map.put("thread1_key" + i, "thread1_value" + i);
                    if (i%10_0000==0) {
                        System.out.println("thread1 put 10W");
                    }
                }
                System.out.println("thread1 end");
                latch.countDown();
            }
        },"thread one").start();
        // 线程2 => t2
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 100_0000; i++) {
                    map.put("thread2_key" + i, "thread2_value" + i);
                    if (i%10_0000==0) {
                        System.out.println("thread2 put 10W");
                    }
                }
                System.out.println("thread2 end");
                latch.countDown();
            }
        },"thread tow").start();
        latch.await();

        //使用JDK1.8
        //没有并发问题应该为:200_0000
        // 实际可能会小于200_0000
        // 实际可能会抛出异常:红黑树的异常,只复现了一次,没有保留结果
        // 实际可能会死循环,用尾插法解决了链表的死循环,但是红黑树还有可能死循环
        //   下面是线程信息
        //   "thread tow" #15 prio=5 os_prio=0 cpu=37656.25ms elapsed=38.05s tid=0x0000018a938c8800 nid=0x3478 runnable  [0x000000066e8fd000]
        //   java.lang.Thread.State: RUNNABLE
        //	   at java.util.HashMap$TreeNode.balanceInsertion(java.base@11.0.5/HashMap.java:2286)
        //	   at java.util.HashMap$TreeNode.treeify(java.base@11.0.5/HashMap.java:1992)
        //	   at java.util.HashMap$TreeNode.split(java.base@11.0.5/HashMap.java:2218)
        //	   at java.util.HashMap.resize(java.base@11.0.5/HashMap.java:709)
        //	   at java.util.HashMap.putVal(java.base@11.0.5/HashMap.java:658)
        //	   at java.util.HashMap.put(java.base@11.0.5/HashMap.java:607)
        //	   at com.spacedo.collocation.Test$2.run(Test.java:35)
        //	   at java.lang.Thread.run(java.base@11.0.5/Thread.java:834)
        //
        //   Locked ownable synchronizers:
        //	- None
        System.out.println(map.size());
    }
}

标签:util,java,HashMap,11.0,链表,JDK8,base,死循环
来源: https://blog.csdn.net/aajjw/article/details/120581508

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有