Java经典八股文之HashMap

简介: 整理HashMap相关八股文

Java相关文章


HashMap原理

内部结构

  1. HashMap内部使用数组+链表(链表长度>8 & 数组大小>64转化为红黑树结构)
  2. hashmap允许key为null但不能重复


为什么要使用红黑树?

  1. 树化是为了hash碰撞严重时链表长度过长,查找复杂度为on
  2. 使用红黑树查询复杂度logn,插入复杂度logn
  3. 根据泊松分布,在负载因子0.75(HashMap默认)的情况下,单个hash槽内元素个数为8的概率小于百万分之一,将7作为一个分水岭,等于7时不做转换,大于等于8才转红黑树,小于等于6才转链表。


为什么不采用AVL树或B树?

  1. 红黑树更通用,在添加、删除、查找的时间复杂度都为logn
  2. AVL树查询快,但添加、删除慢


为什么默认不传值数组大小为16?

  1. 传值初始化大小为大于传值的最小2^n
  2. hashmap的大小始终为2的幂 ,因为计算存放位置时,要将计算出的hash值和hashmap长度-1进行&与运算(同1为1其余都是0),如果是奇数-1最后一位都是0,0&任何数都是0,浪费位数
  3. 取余操作中如果除数是2的幂次则等价于与其除数减一的与操作 (也就是说 hash%length==hash&(length-1)的前提是 length 是2的 n 次方;)。” 并且采用二进制位操作 ,相对于取余操作能够提高运算效率,这就解释了 HashMap 的长度为什么是2的幂次方。


为什么扩容因子是0.75

  1. 符合泊松分布
  2. 扩容因子太大hash冲突会频繁,扩容因子太小空间浪费,查询效率会底
  3. 0.75刚刚好


put原理

  1. 对key的hashcode进行hash计算得到下标
  2. 判断是否存在hash碰撞
  3. 如果碰撞了以链表的形式放在bucket里
  4. 如果链表长度过长(大于默认值8),则把链表转换成红黑树
  5. 如果节点存在则替换value
  6. 如果数组长度大于了 当前容量*负载因子则进行resize


hash运算

  1. hash方法实际是让key.hashCode()与key.hashCode()>>>16进行异或操作
  2. 扰动函数降低hash碰撞几率


get原理

  1. 对key的hashCode()做hash运算,计算index;
  2. 如果在bucket⾥的第⼀个节点⾥直接命中,则直接返回;
  3. 如果有冲突,则通过key.equals(k)去查找对应的Entry;
  4. 若为树,则在树中通过key.equals(k)查找,O(logn);
  5. 若为链表,则在链表中通过key.equals(k)查找,O(n)。


扩容(resize)原理

  1. 每次扩容都为原来的2倍
  2. 扩展后 Node 对象的位置要么在原位置,要么移动到原偏移量两倍的位置
  3. 1.7 ,扩容之后需要重新去计算其 Hash 值,根据 Hash 值对其进行分发.
  4. 1.8 ,则是根据在同一个桶的位置中进行判断(e.hash & oldCap)是否为 0,0 -表示还在原来位置,否则就移动到原数组位置 + oldCap。
  5. 重新进行 hash 分配后,该元素的位置要么停留在原始位置,要么移动到原始位置+增加的数组大小这个位置上。


rehash源码

void transfer(Entry[] newTable) {
    Entry[] src = table;                   //src引用了旧的Entry数组
    int newCapacity = newTable.length;
    for (int j = 0; j < src.length; j++) { //遍历旧的Entry数组
        Entry<K, V> e = src[j];             //取得旧Entry数组的每个元素
        if (e != null) {
            src[j] = null;//释放旧Entry数组的对象引用(for循环后,旧的Entry数组不再引用任何对象)
            do {
                Entry<K, V> next = e.next;
                int i = indexFor(e.hash, newCapacity); //!!重新计算每个元素在数组中的位置
                e.next = newTable[i]; //标记[1]
                newTable[i] = e;      //将元素放在数组上
                e = next;             //访问下一个Entry链上的元素
            } while (e != null);
        }
    }
}


为什么线程不安全

  1. 扩容时,table数组是线程共享的,newtable是线程不共享的
  2. transfer函数执行完会执行table = newtable其他线程就可以看到转移线程转移后的结果了
  3. jdk1.7之前使用头插法导致扩容后数组反转,多线程下产生环、数据覆盖
  4. 产生环的原因
  1. 一是头插法
  2. 二是Java内存模型导致多线程下当被另一个线程执行完扩容后,新数组都是头插法执行后的逆序状态。没及时更新主存数据
相关文章
|
6月前
|
存储 安全 Java
2025 年一线互联网大厂最新高质量 Java 面试八股文整理及答案汇总
本文整理了一线互联网大厂最新的高质量Java面试八股文及其答案,涵盖Java基础、集合框架与多线程三大核心模块。内容包括面向对象与面向过程的区别、`equals`与`==`的差异、`final`和`static`的用法、集合类如`ArrayList`与`LinkedList`的对比、`HashMap`的工作原理及其与`Hashtable`的区别,以及多线程中的线程创建方式、生命周期、上下文切换和死锁等知识点。通过系统化的梳理与解析,帮助读者高效备考Java面试,掌握核心技术要点。资源可从文末链接下载。
1069 40
|
5月前
|
存储 算法 安全
JAVA 八股文全网最详尽整理包含各类核心考点助你高效学习 jAVA 八股文赶紧收藏
本文整理了Java核心技术内容,涵盖Java基础、多线程、JVM、集合框架等八股文知识点,包含面向对象特性、线程创建与通信、运行时数据区、垃圾回收算法及常用集合类对比,附有代码示例与学习资料下载链接,适合Java开发者系统学习与面试准备。
1196 0
|
12月前
|
存储 安全 Java
Java 集合框架中的老炮与新秀:HashTable 和 HashMap 谁更胜一筹?
嗨,大家好,我是技术伙伴小米。今天通过讲故事的方式,详细介绍 Java 中 HashMap 和 HashTable 的区别。从版本、线程安全、null 值支持、性能及迭代器行为等方面对比,帮助你轻松应对面试中的经典问题。HashMap 更高效灵活,适合单线程或需手动处理线程安全的场景;HashTable 较古老,线程安全但性能不佳。现代项目推荐使用 ConcurrentHashMap。关注我的公众号“软件求生”,获取更多技术干货!
216 3
Java之HashMap详解
本文介绍了Java中HashMap的源码实现(基于JDK 1.8)。HashMap是基于哈希表的Map接口实现,允许空值和空键,不同步且线程不安全。文章详细解析了HashMap的数据结构、主要方法(如初始化、put、get、resize等)的实现,以及树化和反树化的机制。此外,还对比了JDK 7和JDK 8中HashMap的主要差异,并提供了使用HashMap时的一些注意事项。
403 2
Java之HashMap详解
|
6月前
|
存储 安全 Java
2025 年一线互联网大厂最新高质量 Java 面试八股文整理带答案及实操要点
本文整理了一线互联网大厂最新的高质量Java面试八股文及答案,涵盖Java基础、集合、多线程等多个核心方面,帮助你高效备考。内容包括面向对象与面向过程的区别、`equals`与`==`的对比、`final`和`static`的用法,以及ArrayList与LinkedList的区别、HashMap的工作原理等。同时,深入探讨了多线程创建方式、生命周期、上下文切换及死锁问题,并附有实操代码示例。资源链接:[点此下载](https://pan.quark.cn/s/14fcf913bae6)。
1155 1
|
6月前
|
存储 安全 Java
Java 集合面试题从数据结构到 HashMap 源码剖析详解及长尾考点梳理
本文深入解析Java集合框架,涵盖基础概念、常见集合类型及HashMap的底层数据结构与源码实现。从Collection、Map到Iterator接口,逐一剖析其特性与应用场景。重点解读HashMap在JDK1.7与1.8中的数据结构演变,包括数组+链表+红黑树优化,以及put方法和扩容机制的实现细节。结合订单管理与用户权限管理等实际案例,展示集合框架的应用价值,助你全面掌握相关知识,轻松应对面试与开发需求。
334 3
|
存储 Java
Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。
【10月更文挑战第19天】本文详细介绍了Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。HashMap以其高效的插入、查找和删除操作著称,而TreeMap则擅长于保持元素的自然排序或自定义排序,两者各具优势,适用于不同的开发场景。
148 1
|
存储 安全 Java
Java Map新玩法:探索HashMap和TreeMap的高级特性,让你的代码更强大!
【10月更文挑战第17天】Java Map新玩法:探索HashMap和TreeMap的高级特性,让你的代码更强大!
260 2
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
284 2