热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

JDK7,8,JD9的hashmap,hashtable,concurrenthashmap及他们的区别

1:hashmap简介(如下,数组-链表形式)HashMap的存储结构图中,紫色部分即代表哈希表,也称为哈希数组(默认数组大小是16,每对key-value键值对其实是存在m

 1:hashmap简介(如下,数组-链表形式)

  HashMap的存储结构

  

  图中,紫色部分即代表哈希表,也称为哈希数组(默认数组大小是16,每对key-value键值对其实是存在map的内部类entry里的),数组的每个元素都是一个单链表的头节点,跟着的绿色链表是用来解决冲突的,如果不同的key映射到了数组的同一位置处,就会采用头插法将其放入单链表中。

  2:hashmap原理(即put和get原理)

  2.1 put原理

  1.根据key获取对应hash值:int hash = hash(key.hash.hashcode())

  2.根据hash值和数组长度确定对应数组引inti=indexFor(hash,table.length);简单理解就是i = hash值%模以数组长度(其实是按位与运算)。如果不同的key都映射到了数组的同一位置处,就将其放入单链表中。且新来的是放在头节点。

  2.2 get原理

  1.通过hash获得对应数组位置,遍历该数组所在链表(key.equals())

  3.1:hashcode相同,冲突怎么办?

  “头插法”,放到对应的链表的头部。

  3.2:为什么是头插法(为什么这么设计)?

  因为HashMap的发明者认为,后插入的Entry被查找的可能性更大,所以放在头部(因为get()查询的时候会遍历整个链表)。

  4.1:hashmap的默认数组长度是多少?

  默认为16

  4.2:为什么?

  之所以选择16,是为了服务于从key映射到index的hash算法(看下面)。

  5.1:hashmap达到默认负载因子(0.75)怎么办?

  自动双倍扩容,扩容后重新计算每个键值对位置。且长度必须为16或者2的幂次

  5.2:为啥要16或者2的幂次?

  若不是16或者2的幂次,位运算的结果不够均匀分布,显然不符合Hash算法均匀分布的原则。

  反观长度16或者其他2的幂,Length-1的值是所有二进制位全为1,这种情况下,index的结果等同于HashCode后几位的值。只要输入的HashCode本身分布均匀,Hash算法的结果就是均匀的。

  6.1:hashmap是线程安全的吗?

  不是。

  6.2 :为什么?

  因为没加锁

  6.3: 那在并发时会导致什么问题?

  hashmap在接近临界点时,若此时两个或者多个线程进行put操作,都会进行resize(扩容)和ReHash(为key重新计算所在位置),而ReHash在并发的情况下可能会形成链表环。在执行get的时候,会触发死循环,引起CPU的100%问题。

  注:jdk8已经修复hashmap这个问题了,jdk8中扩容时保持了原来链表中的顺序。但是HashMap仍是非并发安全,在并发下,还是要使用ConcurrentHashMap。

  6.4: 如何判断有环形表?

  最优:首先创建两个指针A和B(在java里就是两个对象引用),同时指向这个链表的头节点。然后开始一个大循环,在循环体中,让指针A每次向下移动一个节点,让指针B每次向下移动两个节点,然后比较两个指针指向的节点是否相同。如果相同,则判断出链表有环,如果不同,则继续下一次循环。

  理解例子:在一个环形跑道上,两个运动员在同一地点起跑,一个运动员速度快,一个运动员速度慢。当两人跑了一段时间,速度快的运动员必然会从速度慢的运动员身后再次追上并超过,原因很简单,因为跑道是环形的。

  

  7: hashmap 和 hashtable 区别?

  两者的区别线程效率数组默认值null值hashmap不安全更高16key-value都允许hashtable安全略低11不允许(抛异常)

  8.0:那hashmap不安全,hashtable性能又低,怎么办?

  用concurrenthashmap,即保证安全,性能又可以保证。

  8.1:那concurrenthashmap究竟是什么?

  整个ConcurrentHashMap的结构如下:

  

  理解:hashmap是有entry数组组成,而concurrenthashmap则是Segment数组组成。而Segment又是什么呢?Segment本身就相当于一个HashMap。

  同HashMap一样,Segment包含一个HashEntry数组,数组中的每一个HashEntry既是一个键值对,也是一个链表的头节点。

  单一的Segment结构如下(是不是看着就是hashmap):

  

  像这样的Segment对象,在ConcurrentHashMap集合中有多少个呢?有2的N次方个,共同保存在一个名为segments的数组当中。

  可以说,ConcurrentHashMap是一个二级哈希表。在一个总的哈希表下面,有若干个子哈希表。(这样类比理解多个hashmap组成一个cmap)

  8.2:那他的put和get方法呢?

  Put方法:

  1.为输入的Key做Hash运算,得到hash值。

  2.通过hash值,定位到对应的Segment对象

  3.获取可重入锁

  4.再次通过hash值,定位到Segment当中数组的具体位置。

  5.插入或覆盖HashEntry对象。

  6.释放锁。

  Get方法:

  3.再次通过hash值,定位到Segment当中数组的具体位置。

  由此可见,和hashmap相比,ConcurrentHashMap在读写的时候都需要进行二次定位。先定位到Segment,再定位到Segment内的具体数组下标。

  9: hashmap 和 concurrenthashmap区别?

  线程: 不安全 安全

  10.1:为啥concurrenthashmap和hashtable都是线程安全,但是前者性能更高

  因为前者是用的分段锁,根据hash值锁住对应Segment对象,当hash值不同时,使其能实现并行插入,效率更高,而hashtable则会锁住整个map。

  如何理解并行插入:当cmap需要put元素的时候,并不是对整个map进行加锁,而是先通过hashcode来知道他要放在那一个分段(Segment对象)中,然后对这个分段进行加锁,所以当多线程put的时候,只要不是放在同一个分段中,就实现了真正的并行的插入。

  但是,在统计size的时候,就是获取concurrenthashmap全局信息的时候,就需要获取所有的分段锁才能统计(即效率稍低)。

  10.2:分段锁的设计解决的是什么问题?

  分段锁的设计目的是细化锁的粒度,当操作不需要更新整个数组的时候,就仅仅针对数组中的一部分行加锁操作。

  11:JDK1.7的hashmap和JDK1.8的hashmap的区别(即1.8做了哪些优化)?

  1.为了加快查询效率,java8的hashmap引入了红黑树结构,当数组长度大于默认阈值64时,且当某一链表的元素>8时,该链表就会转成红黑树结构,查询效率更高。(问题来了,什么是红黑树?什么是B+树?(mysql索引有B+树索引)什么是B树?什么是二叉查找树?)数据结构方面的知识点会更新在【数据结构专题】,这里不展开。

  这里只简单的介绍一下红黑树:

  红黑树是一种自平衡二叉树,拥有优秀的查询和插入/删除性能,广泛应用于关联数组。对比AVL树,AVL要求每个结点的左右子树的高度之差的绝对值(平衡因子)最多为1,而红黑树通过适当的放低该条件(红黑树限制从根到叶子的最长的可能路径不多于最短的可能路径的两倍长,结果是这个树大致上是平衡的),以此来减少插入/删除时的平衡调整耗时,从而获取更好的性能,而这虽然会导致红黑树的查询会比AVL稍慢,但相比插入/删除时获取的时间,这个付出在大多数情况下显然是值得的。好了我知道你们看晕了,移步去看看我的【数据结构专题】吧。

  2.优化扩容方法,在扩容时保持了原来链表中的顺序,避免出现死循环

  12:JDK1.7的concurrenthashmap和JDK1.8又有什么区别?

  1.8的实现已经抛弃了Segment分段锁机制,利用Node数组+CAS+Synchronized来保证并发更新的安全,底层采用数组+链表+红黑树的存储结构。

  

  java给我们带来了并发安全的ConcurrentHashMap,它的实现是依赖于 Java 内存模型,所以我们在了解 ConcurrentHashMap 的之前必须了解一些底层的知识:

  java内存模型

  java中的Unsafe

  java中的CAS

  java同步器AQS

  ReentrantLock

  所以在这里我不准备深入讲解ConcurrentHashMap ,我会在【并发编程】专题通过一步步详解并发基础,从java内存模型,synchronized,volatile,Unsafe到CAS,AQS,各种锁再到JUC并发包相关。

  先放张java内存模型的思维导图勾引一波,光java内存模型一个点就有这么多要讲的了。

  13:那么问题来了,什么是CAS?

  关于CAS方面的知识点,又会涉及到ABA问题,又可以扯到乐观锁悲观锁,锁编程,AQS等,相关内容将更新在【并发编程专题】,这里不做展开

  

  14:那1.9的呢?

  瞄了一眼,好像和1.8的没啥区别,这里不做展开....(嘤嘤嘤,别打脸.)


推荐阅读
  • HashTable与ConcurrentHashMap均可实现HashMap的功能,对外提供了键值对存储的数据结构。但是在内部结构及实现上有何区别,性能上的差异到底在哪里又是如何导致的 ... [详细]
  • 要讨论这些常用的默认初始容量和扩容的原因是:当底层实现涉及到扩容时,容器或重新分配一段更大的连续内存(如果是离散分配则不需要重新分配,离散分配都是插入新元素时动态分配内存),要将容器原来的数据全部复 ... [详细]
  • HashMap的相关问题及其底层数据结构和操作流程
    本文介绍了关于HashMap的相关问题,包括其底层数据结构、JDK1.7和JDK1.8的差异、红黑树的使用、扩容和树化的条件、退化为链表的情况、索引的计算方法、hashcode和hash()方法的作用、数组容量的选择、Put方法的流程以及并发问题下的操作。文章还提到了扩容死链和数据错乱的问题,并探讨了key的设计要求。对于对Java面试中的HashMap问题感兴趣的读者,本文将为您提供一些有用的技术和经验。 ... [详细]
  • HashMap及HashTable源码解析HashMap在java和Android经常使用到,之前学过数据结构,理解了它的原理,却很 ... [详细]
  • 本文由编程笔记#小编为大家整理,主要介绍了源码分析--ConcurrentHashMap与HashTable(JDK1.8)相关的知识,希望对你有一定的参考价值。  Concu ... [详细]
  • 一面自我介绍对象相等的判断,equals方法实现。可以简单描述挫折,并说明自己如何克服,最终有哪些收获。职业规划表明自己决心,首先自己不准备继续求学了,必须招工作了。希望去哪 ... [详细]
  • 集合类中只能存放对象,而不能存放原始数据类型的元素,所以当有原始数据类型需要存放时,只能将其转换成相应的包装类对象。1)访问和遍历数组元素时,ArrayList的 ... [详细]
  • 一、HashMap1.HashMap概述:HashMap是基于哈希表的Map接口的非同步实现。此实现提供所有可选的映射操作,并允许使用null值和null键。此类不保证映射的顺序,特别是 ... [详细]
  • HashMap和Hashtable的区别主要的区别有三点:线程安全性,同步(synchronization),以及速度。(两者都是无序排放)HashMap几乎可以等价于Hashtable,除了Hash ... [详细]
  • (2.1.8)Java之集合类:set、list、hashmap、hashtable等和迭代器iterator
    一、容器常见的集合类有这些种:实现Collection接口的:Set、List以及他们的实现类。实现Map接口的:HashMap及其实现类编程爱好者学习,下面我我们通过一个图来整体描述一下:这个图片没 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 深入理解Java虚拟机的并发编程与性能优化
    本文主要介绍了Java内存模型与线程的相关概念,探讨了并发编程在服务端应用中的重要性。同时,介绍了Java语言和虚拟机提供的工具,帮助开发人员处理并发方面的问题,提高程序的并发能力和性能优化。文章指出,充分利用计算机处理器的能力和协调线程之间的并发操作是提高服务端程序性能的关键。 ... [详细]
  • ejava,刘聪dejava
    本文目录一览:1、什么是Java?2、java ... [详细]
  • 常用API-Hashtable类及其与HashMap、HashSet的区别转载请表明出处:http:blog.csdn.netu012637501(嵌入式_小J的天空)一、Hashtable&l ... [详细]
  • 最近无聊,用BCB多线程操作一下数据库,没想到开两个线程,做查询操作和插入操作,点击得快一点就出现占线错误,当场晕倒,一看就是查询问题,翻了本网站的所有记录,解决方法有几个,就是不能解决!JAVA里面 ... [详细]
author-avatar
a7431555
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有