集合框架——HashTable和HashMap的区别[通俗易懂]

集合框架——HashTable和HashMap的区别[通俗易懂]继承类不同HashTable继承Dictionary类,HashMap继承AbstractMap类线程安全级别不同HashTable是线程安全的类,每个public方法都有Synchronized修饰,HashMap不是线程安全的是否允许null的要求不同HashTable:key不允许为null,value不允许为nullHashMap:key允许为null,value允许为n…

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

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

  • 继承类不同

HashTable继承Dictionary类,HashMap继承AbstractMap类

  • 线程安全级别不同

HashTable是线程安全的类,每个public方法都有Synchronized修饰,HashMap不是线程安全的

  • 是否允许null的要求不同

HashTable:key不允许为null,value不允许为null

HashMap:key允许为null,value允许为null

  • 底层数据结构不同

在JDK1.8以后,HashMap的底层数据结构改成了数组+链表+红黑树的实现,在链表的节点大于TREEIFY_THRESHOLD=8时,链表转为红黑树,在树节点小于UNTREEIFY_THRESHOLD=6时,红黑树转变为链表。(之所以有8和6两个阈值是为了避免某个链表在临界点频繁插入删除,导致转换频繁降低性能)

而Hashtable的底层实现就是数组+链表,而没有红黑树,因此各种操作都要简单很多。

  • 容量的要求不同

Hashtable的容量是直接使用用户输入的容量initialCapacity。

HashMap在用户输入的基础上,强制将容量转换为大于输入容量的最小2的幂次方数值,通过tableSizeFor(int cap)函数实现。

  • index的计算方法不同

HashTbale是古老的除留余数法,直接使用hashcode

int hash = key.hashCode();  
int index = (hash & 0x7FFFFFFF) % tab.length; 

而HashMap是强制容量为2的幂,重新根据hashcode计算hash值,在使用hash & (length-1),也等价取膜,但更加高效,取得的位置更加分散,偶数,奇数保证了都会分散到,HashTbale就不能保证

static int indexFor(int h, int length) {  
  return h & (length-1);  
  • 扩容方法不同

HashTable扩容后新容量是原容量的2倍+1,新threshold为新容量*loadFactor。然后在oldMap中对数组从尾部开始遍历,对每个链表从头部开始遍历取出节点,重新计算节点在newMap中的hashCode,放进newMap中,并把oldMap中的节点指向null。

HashMap扩容后新容量是原容量的2倍,新threshold为2倍旧threshold。在扩容时,由于是2倍扩容,可以得到新的index要么不变,要么是旧index+oldCapacity。并且是从链表(红黑树)头部开始遍历,并将节点分别顺序放到高、低两个链表中,然后将链表头部链接到数组的相应bucket中。

基于上述不同,因此hashMap的扩容效率更高,并且JDK1.8后也不会出现resize()导致的线程不安全。

  • 方法不同

HashMap把Hashtable的contains方法去掉了,改成containsvalue和containsKey,因为contains方法容易让人引起误解。而Hashtable有contains方法、containsvalue方法和containsKey方法,其中contains方法和containsvalue方法是一样的。

  • 迭代器不同

HashMap的迭代器(Iterator)是fail-fast迭代器,所以当有其它线程改变了HashMap的结构(增加或者移除元素),将会抛出ConcurrentModificationException,但迭代器本身的remove()方法移除元素则不会抛出ConcurrentModificationException异常。

而Hashtable除了有Iterator迭代器还有enumerator迭代器,并且enumerator迭代器不是fail-fast的。

*关于fail-fast机制:fail-fast是通过modcount参数实现的,在HashMap中每当有会修改HashMap结构的操作被执行,那么modcount加1,modcount是volatile的,因此是线程可见的,在迭代过程中,判断modCount跟expectedModCount是否相等,如果不相等就表示已经有其他线程修改了Map。而HashTable并没有这个参数,因此没有fail-fast机制。

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

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

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


相关推荐

  • Android Bundle类

    Android Bundle类

    2021年12月4日
    55
  • Activity工作流–工作原理「建议收藏」

    Activity工作流–工作原理「建议收藏」1.什么是Activity?即工作流,和请假的流程大概相识,不过既然为一种技术,肯定有其用处和运行原理。Activity是Android的四大组件之一。是用户操作的可视化界面;它为用户提供了一个完成操作指令的窗口。当我们创建完毕Activity之后,需要调用setContentView()方法来完成界面的显示;以此来为用户提供交互的入口。在AndroidApp中只要能看见的几乎都要依托于Act…

    2022年7月27日
    5
  • 卡巴斯基kav为什么总提示程序已经断开连接「建议收藏」

    卡巴斯基kav为什么总提示程序已经断开连接「建议收藏」用着用着就出现了程序已经断开连接的提示.要求重启产品.请问为什么会这样?该如何解决?这是KIS的一个BUG,我也遇到过。与楼上说的授权啊地震啊都没有关系。下载新版本的KIS就能解决。http://www.kpfans.com/bbs/index.php这是卡巴斯基爱好者论坛,里面有几乎有所有的新旧版本。 

    2022年8月20日
    7
  • 遗传算法的优化算法_粒子群算法和遗传算法的区别

    遗传算法的优化算法_粒子群算法和遗传算法的区别与遗传算法的第一次接触遗传算法的基本概念基本定义遗传算法的基本流程遗传算法过程中的具体操作参数的编码二进制编码Gray编码实数编码有序编码初始群体的设定适应度函数的计算遗传操作设计选择selection交叉crossover变异mutation控制参数的设定求解优化问题的实例问题描述问题分析算法设计个体编码适应值函数选择策略杂交算子变异算子参数设置

    2025年9月16日
    5
  • FreeLink开源呼叫中心设计思想

    FreeLink开源呼叫中心设计思想

    2022年1月24日
    42
  • 脚本是什么?[通俗易懂]

    脚本是什么?[通俗易懂]初次接触“脚本”一词并不知道这一听似非常高大上的东西是什么,尔后逐渐接触,虽有了解,但也没有仔细地总结和思考过,今日百度了一下,在此小小总结。“脚本”其实就是一段代码,一个程序。这与我们学习C语言时,写的第一个“helloworld”显示程序没有太大的区别,那为什么这个向程序之神打招呼的“helloworld”程序我们不称其为脚本呢?因为“脚本”有这些特别之处:1、脚本语法比较简单…

    2025年7月26日
    3

发表回复

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

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