问题 :

  • HashMap 容量大小 (capacity)为什么为 2n
  • HashMap 是线程安全的吗,为什么
  • HashMap 既然有hash进行排位还需要equals()作用是什么

  文章部分图片和代码来自参考资料,属于半原创

概述

HashMap 属于字典类,以键值对的方式存储值, 通过计算 hash 值,把key 放在特定的位置,当计算得到的键相同将会以链表的形式在冲突点链接, java 8 中当链表长度达到一定数量,该链表会形成一个红黑树,加快查找。所以java8 的HashMap内部的数据结构就成了 “数组+链表+红黑树” (图片出处见参考资料)

源码解析

主要我们关注的的是get 和 put 方法

put 方法

  1. public V put(K key, V value) {
  2. return putVal(hash(key), key, value, false, true);
  3. }
  4.  
  5. public V put(K key, V value) {
  6. return putVal(hash(key), key, value, false, true);
  7. }
  8.  
  9. // 第三个参数 onlyIfAbsent 如果是 true,那么只有在不存在该 key 时才会进行 put 操作
  10. final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
  11. boolean evict) {
  12. Node<K,V>[] tab; Node<K,V> p; int n, i;
  13. // 第一次 put 值的时候,会触发下面的 resize(),初始化数组长度
  14. // 第一次 resize 和后续的扩容有些不一样,因为这次是数组从 null 初始化到默认的 16 或自定义的初始容量
  15. if ((tab = table) == null || (n = tab.length) == 0)
  16. n = (tab = resize()).length;
  17. // 找到具体的数组下标,如果此位置没有值,那么直接初始化一下 Node 并放置在这个位置就可以了
  18. if ((p = tab[i = (n - 1) & hash]) == null)
  19. tab[i] = newNode(hash, key, value, null);
  20.  
  21. else {// 数组该位置有数据
  22. Node<K,V> e; K k;
  23. // 首先,判断该位置的第一个数据和我们要插入的数据,key 是不是"相等"或是这个地方还没插入数值
  24. //,如果是,取出这个节点
  25. if (p.hash == hash &&
  26. ((k = p.key) == key || (key != null && key.equals(k))))
  27. e = p;
  28. // 如果该节点是代表红黑树的节点,调用红黑树的插值方法,本文不展开说红黑树
  29. else if (p instanceof TreeNode)
  30. e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
  31. else {
  32. // 到这里,说明数组该位置上是一个链表
  33. for (int binCount = 0; ; ++binCount) {
  34. // 插入到链表的最后面
  35. if ((e = p.next) == null) {
  36. p.next = newNode(hash, key, value, null);
  37. // TREEIFY_THRESHOLD 为 8,所以,如果新插入的值是链表中的第 8 个
  38. // 会触发下面的 treeifyBin,也就是将链表转换为红黑树
  39. if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
  40. treeifyBin(tab, hash);
  41. break;
  42. }
  43. // 如果在该链表中找到了"相等"的 key(== 或 equals)
  44. if (e.hash == hash &&
  45. ((k = e.key) == key || (key != null && key.equals(k))))
  46. // 此时 break,那么 e 为链表中[与要插入的新值的 key "相等"]的 node
  47. break;
  48. p = e;
  49. }
  50. }
  51. // e!=null 说明存在旧值的key与要插入的key"相等"
  52. // 对于我们分析的put操作,下面这个 if 其实就是进行 "值覆盖",然后返回旧值
  53. if (e != null) {
  54. V oldValue = e.value;
  55. if (!onlyIfAbsent || oldValue == null)
  56. e.value = value;
  57. afterNodeAccess(e);
  58. return oldValue;
  59. }
  60. }
  61. ++modCount;
  62. // 如果 HashMap 由于新插入这个值导致 size 已经超过了阈值,需要进行扩容
  63. if (++size > threshold)
  64. resize();
  65. afterNodeInsertion(evict);
  66. return null;
  67. }
  68.  
  69. final Node<K,V>[] resize() {
  70. Node<K,V>[] oldTab = table;
  71. int oldCap = (oldTab == null) ? 0 : oldTab.length;
  72. int oldThr = threshold;
  73. int newCap, newThr = 0;
  74. if (oldCap > 0) { // 对应数组扩容
  75. if (oldCap >= MAXIMUM_CAPACITY) {
  76. threshold = Integer.MAX_VALUE;
  77. return oldTab;
  78. }
  79. // 将数组大小扩大一倍
  80. else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY &&
  81. oldCap >= DEFAULT_INITIAL_CAPACITY)
  82. // 将阈值扩大一倍
  83. newThr = oldThr << 1; // double threshold
  84. }
  85. else if (oldThr > 0) // 对应使用 new HashMap(int initialCapacity) 初始化后,第一次 put 的时候
  86. newCap = oldThr;
  87. else {// 对应使用 new HashMap() 初始化后,第一次 put 的时候,初始化数组(容器)
  88. newCap = DEFAULT_INITIAL_CAPACITY;
  89. newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY);
  90. }
  91.  
  92. if (newThr == 0) {
  93. float ft = (float)newCap * loadFactor;
  94. newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ?
  95. (int)ft : Integer.MAX_VALUE);
  96. }
  97. threshold = newThr;
  98.  
  99. // 用新的数组大小初始化新的数组
  100. Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap];
  101. table = newTab; // 如果是初始化数组,到这里就结束了,返回 newTab 即可
  102.  
  103. if (oldTab != null) {
  104. // 开始遍历原数组,进行数据迁移。
  105. for (int j = 0; j < oldCap; ++j) {
  106. Node<K,V> e;
  107. if ((e = oldTab[j]) != null) {
  108. oldTab[j] = null;
  109. // 如果该数组位置上只有单个元素,那就简单了,简单迁移这个元素就可以了
  110. if (e.next == null)
  111. newTab[e.hash & (newCap - 1)] = e;
  112. // 红黑树迁移
  113. else if (e instanceof TreeNode)
  114. ((TreeNode<K,V>)e).split(this, newTab, j, oldCap);
  115. else {
  116. // 这块是处理链表的情况,
  117. // 需要将此链表拆成两个链表,放到新的数组中,并且保留原来的先后顺序
  118. // loHead、loTail 对应一条链表,hiHead、hiTail 对应另一条链表,代码还是比较简单的
  119. Node<K,V> loHead = null, loTail = null;
  120. Node<K,V> hiHead = null, hiTail = null;
  121. Node<K,V> next;
  122. do {
  123. next = e.next;
  124. if ((e.hash & oldCap) == 0) {
  125. if (loTail == null)
  126. loHead = e;
  127. else
  128. loTail.next = e;
  129. loTail = e;
  130. }
  131. else {
  132. if (hiTail == null)
  133. hiHead = e;
  134. else
  135. hiTail.next = e;
  136. hiTail = e;
  137. }
  138. } while ((e = next) != null);
  139. if (loTail != null) {
  140. loTail.next = null;
  141. // 第一条链表
  142. newTab[j] = loHead;
  143. }
  144. if (hiTail != null) {
  145. hiTail.next = null;
  146. // 第二条链表的新的位置是 j + oldCap,这个很好理解
  147. newTab[j + oldCap] = hiHead;
  148. }
  149. }
  150. }
  151. }
  152. }
  153. return newTab;
  154. }

扩容时链表迁移可能有点难理解,可以看这里 :

而有关红黑树的操作可以阅读这篇文章

get 方法

  1. public V get(Object key) {
  2. Node<K,V> e;
  3. return (e = getNode(hash(key), key)) == null ? null : e.value;
  4. }
  5.  
  6. final Node<K,V> getNode(int hash, Object key) {
  7. Node<K,V>[] tab; Node<K,V> first, e; int n; K k;
  8. if ((tab = table) != null && (n = tab.length) > 0 &&
  9. (first = tab[(n - 1) & hash]) != null) {
  10. if (first.hash == hash && // always check first node 检查第一个元素
  11. ((k = first.key) == key || (key != null && key.equals(k))))
  12. return first;
  13. if ((e = first.next) != null) { //判断是链表还是红黑树
  14. if (first instanceof TreeNode)
  15. return ((TreeNode<K,V>)first).getTreeNode(hash, key); //红黑树查找
  16. do { //链表查找
  17. if (e.hash == hash &&
  18. ((k = e.key) == key || (key != null && key.equals(k))))
  19. return e;
  20. } while ((e = e.next) != null);
  21. }
  22. }
  23. return null;
  24. }

总结

HashMap 容量大小 (capacity)为什么为 2n

设计到hash值的计算,hash 最简单的方法就是取余,取余的操作可以用移位来代替(移位相比取余操作提高速度和性能),这关系到一个数学,详情看 hash计算

HashMap 是线程安全的吗,为什么

不安全。因为在扩容的时候迁移数据的操作并非原子操作,同时迭代的时候会产生 fail-fast  ,java7hashmap 会产生环形锁。

HashMap 既然有hash进行排位还需要equals()作用是什么

       思考这个问题需要知道 hashCode 和 equals 两个方法的作用是什么?

hashCode 方法和equal都是object 方法,hashcode返回的是对象的hash值,一般返回的是对象的引用地址。equals 这是逻辑上判断两个对象是不是相同的,例如在HashMap 中假如某个位置已经有一个 n1<“s”,3> , 现在要更新的传进来的是<”s”,5>,那么这个”s“在逻辑上是相同的东西吗?很明显这两个对象是在逻辑应该认同是相同的。通常有以下原则 :

  • 两对象equals相等(逻辑上认为是相同的),那么hashCode 也必然要是相等
  • equals 不相等,hashcode有可能一样也有可能不一样。
  • 为了保证第一条原则,就要求我们在要是重写了equals 方法,那么就要重写 hashCode方法。

综上,使用hash进行排位后还需使用equals的原因是: hash相等和equals 相等后可以从逻辑上确定这两者是相同的东西,例如String 重写了 hashCode 和 equals 方法。

下面看一下这几个类的 equals 方法 和 hashcode 方法

  1. //Object 中的hashCode 方法 和 equals 方法
  2.  
  3. //可以看到hashcode 的实现不由 java提供,源码注解中也有说明
  4. public native int hashCode();
  5.  
  6. //比较一下引用地址
  7. public boolean equals(Object obj) {
  8. return (this == obj);
  9. }
  10.  
  11. //String 重写了这两个方法
  12.  
  13. public boolean equals(Object anObject) {
  14. if (this == anObject) {
  15. return true;
  16. }
  17. if (anObject instanceof String) {
  18. String anotherString = (String)anObject;
  19. int n = value.length;
  20. if (n == anotherString.value.length) {
  21. char v1[] = value;
  22. char v2[] = anotherString.value;
  23. int i = 0;
  24. while (n-- != 0) {
  25. if (v1[i] != v2[i])
  26. return false;
  27. i++;
  28. }
  29. return true;
  30. }
  31. }
  32. return false;
  33. }
  34.  
  35. public int hashCode() {
  36. int h = hash;
  37. if (h == 0 && value.length > 0) {
  38. char val[] = value;
  39.  
  40. for (int i = 0; i < value.length; i++) {
  41. h = 31 * h + val[i];
  42. }
  43. hash = h;
  44. }
  45. return h;
  46. }
  47.  
  48. //hashMap中定义的节点(内部类 Node)也重写了这两个方法
  49.  
  50. public final boolean equals(Object o) {
  51. if (o == this)
  52. return true;
  53. if (o instanceof Map.Entry) {
  54. Map.Entry<?,?> e = (Map.Entry<?,?>)o;
  55. if (Objects.equals(key, e.getKey()) &&
  56. Objects.equals(value, e.getValue()))
  57. return true;
  58. }
  59. return false;
  60. }
  61.  
  62. public final int hashCode() {
  63. return Objects.hashCode(key) ^ Objects.hashCode(value);
  64. }
  65.  
  66. //Objects.equals 的方法
  67.  
  68. public static boolean equals(Object a, Object b) {
  69. return (a == b) || (a != null && a.equals(b));
  70. }

HashSet

说到 HashMap就有必要说一下 HashSet ,为什么呢?

  1. private transient HashMap<E,Object> map;
  2.  
  3. public HashSet() {
  4. map = new HashMap<>();
  5. }
  6.  
  7. public boolean add(E e) {
  8. return map.put(e, PRESENT)==null;
  9. }
  10.  
  11. public boolean remove(Object o) {
  12. return map.remove(o)==PRESENT;
  13. }

对于HashSet 我们需要知道 :

  • HashSet 实际上是调用 HashMap 的方法
  • 它不允许集合中出现重复元素
  • HashSet 是线程不安全的

参考资料

java 基础 --- java8 HashMap的更多相关文章

  1. Java基础系列--HashMap(JDK1.8)

    原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/10022092.html Java基础系列-HashMap 1.8 概述 HashMap是 ...

  2. 【Java基础】HashMap原理详解

    哈希表(hash table) 也叫散列表,是一种非常重要的数据结构,应用场景及其丰富,许多缓存技术(比如memcached)的核心其实就是在内存中维护一张大的哈希表,本文会对java集合框架中Has ...

  3. java基础之hashmap

    Hashmap是一种非常常用的.应用广泛的数据类型,最近研究到相关的内容,就正好复习一下.网上关于hashmap的文章很多,但到底是自己学习的总结,就发出来跟大家一起分享,一起讨论. 1.hashma ...

  4. 【Java基础】HashMap工作原理

    HashMap Hash table based implementation of the Map interface. This implementation provides all of th ...

  5. Java基础:HashMap中putAll方法的疑惑

    最近回顾了下HashMap的源码(JDK1.7),当读到putAll方法时,发现了之前写的TODO标记,当时由于时间匆忙没来得及深究,现在回顾到了就再仔细思考了下 @Override public v ...

  6. Java基础:HashMap假死锁问题的测试、分析和总结

    前言 前两天在公司的内部博客看到一个同事分享的线上服务挂掉CPU100%的文章,让我联想到HashMap在不恰当使用情况下的死循环问题,这里做个整理和总结,也顺便复习下HashMap. 直接上测试代码 ...

  7. java基础---->java8中的函数式接口

    这里面简单的讲一下java8中的函数式接口,Function.Consumer.Predicate和Supplier. 函数式接口例子 一.Function:接受参数,有返回参数 package co ...

  8. JAVA基础篇—HashMap

    /class Depositor package 银行储户; public class Depositor { private String id; private String name; priv ...

  9. java基础之 hashmap

    Hashmap是一种非常常用的.应用广泛的数据类型,最近研究到相关的内容,就正好复习一下.网上关于hashmap的文章很多,但到底是自己学习的总结,就发出来跟大家一起分享,一起讨论. 1.hashma ...

随机推荐

  1. JAVA中-面向网络编程---单层交互

    面向网络编程---单层交互: 客户端说明: /* * 实现TCP客户端,链接到服务器 * 和服务器实现数据交互 * 实现TCP客户端的类 java.net.Scoket * 构造方法: * Socke ...

  2. Jmeter分布式测试实战

    一.Jmeter分布式测试基础 1.Jmeter分布式测试原因: 在使用Jmeter进行接口的性能测试时,由于Jmeter 是JAVA应用,对负载机的CPU和内存消耗比较大.所以当需要模拟数以万计的并 ...

  3. 学习xss模拟构造攻击(第一篇)

    本文作者:i春秋签约作家——rosectow 0×00前言 XSS又名叫CSS全程(cross site scriptting),中文名跨站脚本攻击,目前网站的常见漏洞之一,它的危害没有像上传漏洞,s ...

  4. 如何查看mysql 默认端口号和修改端口号

    http://blog.itpub.net/26148431/viewspace-1466379/ 1,登录mysql 2,使用命令show global variables like 'port'; ...

  5. Unable to access 'default.path.data' (/var/lib/elasticsearch)

  6. 3.1 High Availability

    摘要: 出处:黑洞中的奇点 的博客 http://www.cnblogs.com/kelvin19840813/ 您的支持是对博主最大的鼓励,感谢您的认真阅读.本文版权归作者所有,欢迎转载,但请保留该 ...

  7. 架构师养成记--30.Redis环境搭建

    Redis的安装 下载地址http://redis.io/download 安装步骤: 首先需要安装gcc,把下载好的redis-3.0.0-rc2.tar.gz 放到 /usr/local 文件夹下 ...

  8. canvas+js绘制折线图

    效果: 源码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="U ...

  9. c#StreamWriter,StreamReader类(主要用于文本文件访问)

    1.为什么要使用StreamReader或者StreamWriter 如果对文本文件需要读取一部分显示一部分则使用FileStream会有问题,因为可能FileStream会在读取的时候把一个汉字的字 ...

  10. Mac OS 10.12后Caps lock(大写键)无法使用的解决办法

    ▲打开设置中的键盘选项,并切换至输入源选项标签, ▲取消勾选“使用大写锁定键来回切换“美国英文””, ▲这时再按下Caps lock即可正常使用大小写切换. ▲Update:目前macOS 10.12 ...