hashmap扩容死锁简书_sql死锁

hashmap扩容死锁简书_sql死锁HashMap扩容HashMap扩容transfer()函数原Entry数组转移到新Entry数组扩容死锁单线程扩容多线程扩容死锁HashMap扩容HashMap在JDK1.7使用的是数组+链表的方式,而在JDK1.8及以后则使用的是数组+链表+红黑树的方式进行数据存储。本文主要是对JDK1.7中存在的死锁问题进行分析。transfer()函数/***TransfersallentriesfromcurrenttabletonewTable.*/v

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

HashMap扩容

HashMap在JDK1.7使用的是数组+链表的方式,而在JDK1.8及以后则使用的是数组+链表+红黑树的方式进行数据存储。本文主要是对JDK1.7中存在的死锁问题进行分析。

transfer()函数

 /** * Transfers all entries from current table to newTable. */
    void transfer(Entry[] newTable, boolean rehash) { 
   
        int newCapacity = newTable.length;
        for (Entry<K,V> e : table) { 
   
            while(null != e) { 
   
                Entry<K,V> next = e.next;//第一行
                if (rehash) { 
   
                    e.hash = null == e.key ? 0 : hash(e.key);
                }
                int i = indexFor(e.hash, newCapacity);//第二行
                e.next = newTable[i];//第三行
                newTable[i] = e;//第四行
                e = next;//第五行
            }
        }
    }

第一行:记录oldhash表中e.next;
第二行:rehash计算数组的位置;
第三行:e要插入到链表的头部,所以要将e.next指向new hash的表中的第一个元素;
第四行:将e放入到newTable当中;
第五行:将e指向下一个节点。

原Entry数组转移到新Entry数组

数组的容量首先是2的指数次方大小,如果无构造参数,默认大小为16。当数组的大小超过扩容阈值的时候,就会扩容,一般扩容为之前的2倍。在JDK1.7中主要使用的是头插法的方式进行数组扩容。从原数组转移数据到新数组时,假设所有数据还是会落在同一索引下,那么同一链表下的数据的存储位置会发生反转,头变成尾,尾变成头。

扩容死锁

单线程扩容

假设:hash的算法就是简单的key与length(数组长度)的求余。hash表的长度为2,如果不扩容,那么元素3,5,7按照计算(key%length)都应该碰撞到table[1]上。
扩容:将hash表的长度将会变成4,然后重新计算hash。
第一步:e指向key(3),next指向key(7);
第二步:计算key(3)将会落在new[3]上面(3%4=3),将key(3)指向new[3],并将key(3)放到new[3];
第三步:e指向key(7);
第四步:next指向key(5);
第五步:计算key(7)将会落在new[3]上面(7%4=3),key(7)的next指向key(3),采用头部插入法,将key(7)插入new[3];
第六步:同上。直到e指向null跳出循环。
在这里插入图片描述

多线程扩容死锁

假设有两个T1、T2线程同时put,同时进入到transfer()。
Entry<K,V> next = e.next;代码块处,线程T2中断,此时T1线程执行完线程扩容,T2继续执行。
第一步:e2和next2当前指向的位置为T1处的值;
第二步::计算key(3)将会落在new2[3]上面(3%4=3),将key(3)指向new2[3],并将key(3)放到new[3],e2和next2指向key(7);
第三步:next2指向key(7).next,也就是T2中的key(3),e2指向T1的key(7);
第四步:将key(7)插入到T2的new[3]位置,根据e2 =next2,此时e2和next2都指向key(3);
第五步:此时next2=e2.next,next2指向null,key(3)重新插入到T2-3中,key(3).next = key(7),此时e2指向null,形成闭环。

在这里插入图片描述
当再有新的key插入T2-3时,首先要判断当前key是否存在时,在for (Entry<K,V> e = table[i]; e != null; e = e.next)就会进入死循环。

public V put(K key, V value) { 
   
        if (table == EMPTY_TABLE) { 
   
            inflateTable(threshold);
        }
        if (key == null)
            return putForNullKey(value);
        int hash = hash(key);
        int i = indexFor(hash, table.length);
        for (Entry<K,V> e = table[i]; e != null; e = e.next) { 
   
            Object k;
            if (e.hash == hash && ((k = e.key) == key || key.equals(k))) { 
   
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }

        modCount++;
        addEntry(hash, key, value, i);
        return null;
    }

JDK1.8对死锁的改进请见:HashMap扩容改进分析

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/191144.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 2020爱分析·智能通讯云厂商全景报告[通俗易懂]

    2020爱分析·智能通讯云厂商全景报告[通俗易懂]报告摘要企业面临业务增长的压力,通讯能力建设成为对内提升运营效率,对外提升产品竞争力和客户体验的重要手段,通讯云的场景应用正在加速渗透。基于对国内各行业甲方企业的调研,爱分析认为智能通讯云应用呈现以下趋势:•在提高内部运营效率方面,企业需要通过强化统一通讯能力,及提高办公智能化的方式实现;•对于企业的客服部门或呼叫中心而言,跑马圈地的时代已经过去,随着企业营销与服务渠道多元化,重心在促进营销转化与提升客户体验;•针对各行业的不同业务场景,通讯能力不仅是音视频交互工具的基础,还需..

    2022年5月18日
    48
  • cmd命令 拷贝某文件夹及其子文件夹文件到其它文件夹

    cmd命令 拷贝某文件夹及其子文件夹文件到其它文件夹

    2022年1月28日
    59
  • U盘重装系统

    U盘重装系统U盘重装系统

    2022年4月24日
    42
  • python–threading多线程总结[通俗易懂]

    python–threading多线程总结[通俗易懂]threading用于提供线程相关的操作,线程是应用程序中工作的最小单元。python当前版本的多线程库没有实现优先级、线程组,线程也不能被停止、暂停、恢复、中断。threading模块提供的类:

    2022年7月3日
    33
  • [CentOS]离线安装gcc/gcc-c++-「建议收藏」

    [CentOS]离线安装gcc/gcc-c++-「建议收藏」环境CentOS7.5官方地址:https://gcc.gnu.org/releases.html镜像站点1:http://mirrors.aliyun.com/centos/7/os/x86_64/Packages/镜像站点2:https://vault.centos.org/7.5.1804/os/x86_64/Packages/gcc:4.8.5安装GCCgcc依赖包清单如下(镜像站下载)cpp-4.8.5-44.el7.x86_64.rpmgcc-4.8.5-44.el7.

    2022年5月9日
    615
  • JAVA对象转map_java处理字符串类型的map

    JAVA对象转map_java处理字符串类型的map1、使用fastJson将String转map:Stringout;ObjectsuccesResponse=JSON.parse(out);//先转换成ObjectMapmap=(Map)succesResponse;//Object强转换为Map2、String转java对象fastjson应用string字符串转换成java对象或者对象数…

    2025年9月4日
    4

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号