HashMap 1.8之后的解读

加入了红黑树,并且对hash函数加以改进

b).每个对象都有一个hashcode,但是hashmap并没有直接使用它,而是使用hash函数进行了一次处理,处理完之后用indexfor求出所在的位置。

对于1.7的hash,纯粹是数学计算,不用理会,可以看1.8的hash优化

c).1.8的的hash优化

 static final int hash(Object key) {
int h; // 下面是将低16位和高16位做个异或运算,高16位保持不变
return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
}

因为key的原始哈希值很大(int类型,32位),而数组很小,所以如果直接用的话,高位信息将会丢失,会增大冲突的可能性。所以java要做一个优化,将key原始的哈希值低16位和高16位做异或运算,不浪费高位资源,使得低位更随机,详细论述参看:

https://www.cnblogs.com/zhengwang/p/8136164.html

https://www.zhihu.com/question/62923854/answer/204445142

d).对于indexfor  

1 static int indexFor(int h, int length) {
2 return h & (length-1);
3 }

   h%length,当length是2的次幂时,等价于h & (length-1),因为length为2的次幂时,length - 1的低位全是1,高位全是0,位与运算保留hash值的低位,即对应的余数.位运算要比模运算要快

  冲突会少,让数据均匀分布,因为16-1的二进制是1111(掩码)低位的四个全部保留,而15-1的二进制1110,只保留了低位的前三位,丢失了哈希值信息,可以参考下面实验

  

  图片来自https://blog.csdn.net/qq_27093465/article/details/52207152

3.常见面试题

https://blog.csdn.net/u012512634/article/details/72735183

好文:http://www.importnew.com/28263.html

1.8详解:https://www.cnblogs.com/yangming1996/p/7997468.html

1.8 hash函数和tableSizeFor的理解:https://blog.csdn.net/fan2012huan/article/details/51097331

理解HashMap:

1.底层结构(数组+ 链表  / 数组+ 链表+ 红黑树  Entry数据结构)

  用红黑树可以加快检索速度

2. put方法:

  -- 解决冲突的方法(拉链法)     其他方法(开放地址法/再散列法)

  -- hashcode()和equals的理解

    https://github.com/CyC2018/CS-Notes/blob/master/notes/Java%20%E5%9F%BA%E7%A1%80.md#hashcode

  -- 理解indexfor:

    1)HashMap的桶的个数为什么是2的幂

    2)为什么要rehash

3.扩容机制以及并发条件下hashmap成环问题

  https://blog.csdn.net/u011305680/article/details/80511885

  -- jdk7 总结

    创建一个新的数组newTable,容量是oldTable的一倍
    遍历oldTable,拿到每个链表
    遍历链表,头插法插入newTable

  -- jdk8 高低位链表

    hash值h二进制为 0010 1100 1111 0001 1110 1110
    旧容量为length=16,二进制为 0001 0000

    两个相与,可以计算出低位第五位的二进制数是否是1,从而确定高低链

  -- 成环问题  因为1.7是头插法,而头插法会导致成环    https://coolshell.cn/articles/9606.html   hashmap1.8不会成环,因为尾插,但并发条件下还是用concurrenthashmap的好

      

4.键值允许为null,默认index=0

5.hashtable,hashmap,concurrenthashmap简要对比https://www.cnblogs.com/heyonggang/p/9112731.html

6.concurrenthashmap好文https://www.cnblogs.com/shan1393/p/9020564.html

7.  1.7和1.8的不同

  -- 底层结构

    1.7 数组加链表; 1.8 是数组加链表,再加红黑树(数量为8转成红黑树,数量为6转成链表)

  -- hash方法

    1.8的更加简单,高位右移十六位再做异或

  -- 扩容机制的不同

    1.8 不会成环(尾插)

·        1.8 不用再次求index(高低链表)

        -------以上是优化

  -- 扩容的条件

    1.7 :size > threshold && bucket[hashIndex] != null

    1.8:size > threshold

  -- 1.8中的扩容包含了初始化

  -- 为什么不用AVL  https://blog.csdn.net/hustyangju/article/details/27214251?utm_source=tuicool

8 concurrenthashmap:

  1.8相当于放弃了segment结构,直接使用数组的第一个对象作为该链表的锁。

HashMap源码解读的更多相关文章

  1. HashMap源码解读(转)

    http://www.360doc.com/content/10/1214/22/573136_78188909.shtml 最近朋友推荐的一个很好的工作,又是面了2轮没通过,已经是好几次朋友内推没过 ...

  2. HashMap源码解读(JDK1.7)

    哈希表(hash table)也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表,而HashMap的实现原理也常常出 ...

  3. jdk 8 HashMap源码解读

    转自:https://www.cnblogs.com/little-fly/p/7344285.html 在原来的作者的基础上,增加了本人对源代码的一些解读. 如有侵权,请联系本人 这几天学习了Has ...

  4. 深入理解JAVA集合系列一:HashMap源码解读

    初认HashMap 基于哈希表(即散列表)的Map接口的实现,此实现提供所有可选的映射操作,并允许使用null值和null键. HashMap继承于AbstractMap,实现了Map.Cloneab ...

  5. hashMap 源码解读理解实现原理和hash冲突

    hashMap 怎么说呢. 我的理解是 外表是一个set 数组,无序不重复 . 每个set元素是一个bean ,存着一对key value 看看代码吧 package test; import jav ...

  6. HashMap源码解读(jdk1.8)

    1.相关常量 默认初始化容量(大小) static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; 最大容量 static final int M ...

  7. HashMap 源码解读

    HashMap在JDK1.7和1.8中有了很大的改变,空闲时间对HashMap做了一点点的研究. HashMap是一种数组和链表结合的数据结构,我们每次new一个HashMap时,都会构造出一个长度为 ...

  8. java集合之HashMap源码解读

    源自:jdk1.8.0_121 HashMap继承自AbstractMap,实现了Map.Cloneable.Serializable. HashMap内部是由数组.链表.红黑树实现的 变量 // 默 ...

  9. JDK容器类Map源码解读

    java.util.Map接口是JDK1.2开始提供的一个基于键值对的散列表接口,其设计的初衷是为了替换JDK1.0中的java.util.Dictionary抽象类.Dictionary是JDK最初 ...

随机推荐

  1. Spring MVC前传递和后端接收的参数名不一致处理方式

    前端传递的变量和后端接收的变量名字不一致时,用注解@RequestParam来实现数据的传递 例如:@RequestParam(value="id") //实现商品的分类目录展现 ...

  2. win10系统jdk安装和环境变量配置

    新换电脑的原因,要重新安装jdk,完整记录一下安装过程 jdk版本用的1.7(公司默认版本) 这是jdk安装目录   更改为D:\jdk\java\jdk1.7 安装jre目录  更改为D:\jdk\ ...

  3. oracle 12c多租户下的日常操作变化

    Oracle 12c创建用户时出现“ORA-65096: invalid common user or role name”的错误 在oracle中,引入了多租户概念,以前是一个instance对应一 ...

  4. tf.nn.relu

    tf.nn.relu(features, name = None) 这个函数的作用是计算激活函数 relu,即 max(features, 0).即将矩阵中每行的非最大值置0. import tens ...

  5. NOIP 车站分级 (luogu 1983 & codevs 3294 & vijos 1851) - 拓扑排序 - bitset

    描述 一条单向的铁路线上,依次有编号为 1, 2, ..., n 的 n 个火车站.每个火车站都有一个级别,最低为 1 级.现有若干趟车次在这条线路上行驶,每一趟都满足如下要求:如果这趟车次停靠了火车 ...

  6. 针对Xcode 9 + iOS11 的修改,及iPhone X的适配

    1,UIScrollView的automaticallyAdjustsScrollViewInsets 失效了. automaticallyAdjustsScrollViewInsets,当设置为YE ...

  7. 16 级高代 II 思考题九的七种解法

    16 级高代 II 思考题九  设 $V$ 是数域 $\mathbb{K}$ 上的 $n$ 维线性空间, $\varphi$ 是 $V$ 上的线性变换, $f(\lambda),m(\lambda)$ ...

  8. fedora23 桌面工具栏fedy, 桌面美化 allow: 2'lau不是2'l2u

    ## Sudoers allows particular users to run various commands as ## the root user, without needing the ...

  9. SpringBoot 整合使用dubbo

    这里主要是按照teaey作者的spring-boot-starter-dubbo框架进行一些变化的使用 依赖包: <dependency> <groupId>com.aliba ...

  10. P3980 [NOI2008]志愿者招募

    思路 巧妙的建图 因为每个志愿者有工作的时段,所以考虑让一个志愿者的流量能够从S流到T产生贡献 所以每个i向i+1连INF-a[x]的边(类似于k可重区间集),每个si向ti连边cap=INF,cos ...