库调多了,都忘了最基础的概念 -HashMap 篇

简介: 库调多了,都忘了最基础的概念 -HashMap 篇

🍁 作者:知识浅谈,CSDN博客专家,阿里云签约博主,InfoQ签约博主,华为云云享专家

📌 擅长领域:全栈工程师、爬虫、ACM算法

💒 公众号:知识浅谈

温馨提醒:由于内容较好,请18岁以上成年人观看

🤞这次都给他拿下🤞

🎈说一下HashMap底层实现?及元素添加流程?

因为JDK1.7和JDK1.8是有区别的,所以按照不同的版本记录

JDK1.7版本:

  1. 底层结构使用数组+链表的方式实现。
  2. 数组中最初的大小为16个节点,阈值为0.75,当达到阈值的时候进行扩容,扩容每次扩容为原来的2倍。
  3. 链表插入节点的时候使用头插法。

😉元素添加的过程:这个版本的元素添加是先判断是否达到阈值,即先扩容,后添加。

添加的时候找到对应的位置,如果为空进行赋值,否则如果是链表采用头插法把值插入。

🧐存在的问题:存在循环链表和值覆盖的问题。

JDK1.8版本:

  1. 底层结构使用数组+链表+红黑树的方式实现。
  2. 数组中最初的大小为16个节点,阈值为0.75,当达到阈值的时候进行扩容,扩容每次扩容为原来的2倍,相比于JDK1.7还有一点就是当链表的节点数大于8且整个map中元素的个数大于64的时候,链表转化为红黑树,当链表中的节点数小于6的时候,红黑树退化为链表。
  3. 链表插入节点的时候使用尾插法。

😉元素添加的过程:这个版本的元素添加是先添加,后判断是否进行扩容。

添加的时候找到对应的位置,如果为空进行赋值,否则如果是链表采用尾部插法把值插入,否则如果是红黑树,则在红黑树中插入对应的节点。

🧐存在的问题: 解决了循环链表的问题,但是仍然存在值覆盖的问题。

🎈为什么HashMap会产生死循环?

这个死循环就是上边提到的循环链表的问题,这个问题是发生在扩容的时候,当多个线程进行节点并发插入的时候,都需要进行扩容,一个线程扩容完,另一个线程本来在前一个线程扩容之前已经指向原本的头节点,扩容之后头节点指向的next节点变化了,当第二个线程扩容的时候就行成了循环。

🎈HashMap除了死循环之外,还有什么问题?

正如上边提到的,除了死循环,还有值覆盖的问题,就是当数组中的一个节点为空的时候两个元素要同时插入的时候当一个节点获取了位置要插入的时候,时间片到了,另一个线程插入了,之后到另一个线程的时候也进行了插入,就把之前的给覆盖了。

🎈为什么ConcurrentHashMap是线程安全的?

📐JDK1.7版本:

使用分段锁的形式,即用segment数组来进行加锁的形式,每个锁下记录一个数组+链表的结构,这个数组的初始值和阈值为16和0.75,同理每个segment下都是如此,为什么concurrentHashMap是安全的,就是因为当修改的时候先锁住一个段下的所有内容进行修改,如果不同段中的数据,是可以并行修改的。

📐JDK1.8版本:

采用Synchronized+CMS的形式,就是对数组中每个节点加synchronized的形式,然后在进行扩容,sychronized锁住结点之后使用CMS的方法进行扩容,并且还支持并发扩容,也就是可以多个线程同时进行扩容,且扩容不用计算hash值,如果之前的hash大于原来的数组则就把当前节点移动到当前节点+原数组长度的位置。

🍚总结

以上就是关于hashmap的简单理解,太深了我也不太理解,希望有所帮助。

相关文章
|
6月前
|
存储 算法 Java
Java集合框架知识点学习核心总结
Java集合框架包含Collection、List(ArrayList、LinkedList)、Set(HashSet、TreeSet)、Map(HashMap、TreeMap)接口及迭代器、泛型、比较器。迭代器用于遍历集合,泛型避免类型转换,比较器用于元素比较。集合框架还提供排序、查找、去重算法。Java 8新增Stream API、Lambda表达式和Optional类,提升集合操作效率。
49 6
|
6月前
|
存储 安全 Java
从源码角度来谈谈 HashMap
HashMap的知识点可以说在面试中经常被问到,是Java中比较常见的一种数据结构。所以这一篇就通过源码来深入理解下HashMap。
79 0
从源码角度来谈谈 HashMap
|
20天前
|
存储 机器学习/深度学习 安全
ConcurrentHashMap核心原理,这次彻底给整明白了!
ConcurrentHashMap核心原理,这次彻底给整明白了!
ConcurrentHashMap核心原理,这次彻底给整明白了!
|
20天前
|
存储 安全 Java
学习集合类源码对我们实际工作的帮助和应用!
学习集合类源码对我们实际工作的帮助和应用!
|
3月前
|
存储 Java
【Java集合类面试二十九】、说一说HashSet的底层结构
HashSet的底层结构是基于HashMap实现的,使用一个初始容量为16和负载因子为0.75的HashMap,其中HashSet元素作为HashMap的key,而value是一个静态的PRESENT对象。
|
3月前
|
安全 Java
【Java集合类面试十五】、说一说HashMap和HashTable的区别
HashMap和Hashtable的主要区别在于Hashtable是线程安全的,不允许null键和值,而HashMap是非线程安全的,允许null键和值。
|
3月前
|
Java
【Java集合类面试十四】、HashMap是如何解决哈希冲突的?
HashMap解决哈希冲突的方法是通过链表和红黑树:当链表长度超过一定阈值时,转换为红黑树以提高性能;当链表长度缩小到另一个阈值时,再转换回链表。
|
3月前
|
Java
【Java集合类面试十九】、说一说你对LinkedHashMap的理解
LinkedHashMap通过维护一个双向链表来保证key-value对的迭代顺序与插入顺序一致,它提供了HashMap的性能和有序迭代的特性,但相比HashMap性能略低。
|
存储 算法 Java
Java集合框架知识点的学习核心总结
Java集合框架是Java中用于存储和操作数据的一组类和接口。以下是Java集合框架知识点的学习核心总结
60 1
|
存储 安全 算法
66.Java容器面试题:谈谈你对 HashMap 的理解
66.Java容器面试题:谈谈你对 HashMap 的理解
168 1