并发编程下的集合:数组寻址、LinkedList、HashMap、ConcurrentHashMap

数组如何寻址

a[n]=起始地址+(n*字节数) 译:a[2]=100+(2*4) 2为下标、4为int类型字段占四个字节

在这里插入图片描述

LinkedList

LinkedList为双向链表结构,链表结构又分为单向、双向、以及循环。

// 双向链表
private static class Node<E> {E item;Node<E> next;Node<E> prev;Node(Node<E> prev, E element, Node<E> next) {this.item = element;this.next = next;this.prev = prev;}}//单向链表private static class Node<E> {E item;Node<E> next;Node(Node<E> prev, E element, Node<E> next) {this.item = element;this.next = next;}}//循环链表private static class Node<E> {E item;Node<E> next;Node<E> prev;Node<E> head;Node<E> tail;Node(Node<E> prev, E element, Node<E> next, Node<E> head, Node<E> tail) {this.item = element;this.next = next;this.prev = prev;this.head=head;this.tail=tail;}}

HashMap

数据结构:数组+链表+红黑树

注意:单个链表的长度达到8时会转换为红黑树,但是删除元素后长度小于6时则会将红黑树转换为链表

在这里插入图片描述

当lies存入后,foes值存入时出现Hash冲突,则用链表存储foes。若lies这个数组节点的链表长度大于8且Hash表大于64时该链表结构转换为红黑树

在这里插入图片描述

链表的插入规则以及问题

JDK1.7头插法

先插入A值后插入B值,在插入B值时出现与A值的Hash冲突,那么B值会替代A值原有的位置,然后将B值的next指向A值。

图一为A值插入后的初始状态

在这里插入图片描述

图二为插入A值后插入B值,B值取代A值的位置,B值next指向A值

在这里插入图片描述

JDK1.8 尾插法,插入A值后插入B值,A值的next指向B。

在这里插入图片描述

如果发现hash取模后的数组索引位下无元素则直接新增,若不是空那就说明存在hash冲突,则判断数组索引位链表结构中的第一个元素的key以及hash值是否与新的key一致则直接覆盖,若不一致则判断当前的数组索引下的链表结构是否为红黑树,若为红黑树则走红黑树的新增方法,若不为红黑树则遍历当前链表结构,遍历中发现某个节点元素的next为null是则直接将新元素指针与next进行关联,若在遍历到next为空前判断到,某个节点的key以及key的hash值与新的key与新的keyhash值一致时则走覆盖。

 final V putVal(int hash, K key, V value, boolean onlyIfAbsent,boolean evict) {Node<K,V>[] tab; Node<K,V> p; int n, i;//扩容HashMapif ((tab = table) == null || (n = tab.length) == 0)n = (tab = resize()).length;//hash是key的hash值取模,如果tab[hash]等于null说明该数组索引下没有值则直接插入if ((p = tab[i = (n - 1) & hash]) == null)tab[i] = newNode(hash, key, value, null);else {Node<K,V> e; K k;//数组索引下链表为一时才会触发 》》如果hash值以及key值一致则覆盖if (p.hash == hash &&((k = p.key) == key || (key != null && key.equals(k))))e = p;//若当前节点为红黑树则通过红黑树的方式将元素添加到树中else if (p instanceof TreeNode)e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);else {//当不是相同key或不是红黑树时遍历当前数组索引的链表结构for (int binCount = 0; ; ++binCount) {//遍历到链表中某个节点的next节点为空则直接将链表最后一个节点的next指针指向新的元素if ((e = p.next) == null) {p.next = newNode(hash, key, value, null);//若发现添加元素后链表长度大于8则将当前数组索引的链表转换为红黑树结构if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1sttreeifyBin(tab, hash);break;}//若发现hash值以及key一致则认为要覆盖if (e.hash == hash &&((k = e.key) == key || (key != null && key.equals(k))))break;p = e;}}//两个覆盖逻辑,若发现一致则覆盖if (e != null) { // existing mapping for keyV oldValue = e.value;if (!onlyIfAbsent || oldValue == null)e.value = value;afterNodeAccess(e);return oldValue;}}++modCount;if (++size > threshold)resize();afterNodeInsertion(evict);return null;}

ConcurrentHashMap

 final V putVal(K key, V value, boolean onlyIfAbsent) {if (key == null || value == null) throw new NullPointerException();int hash = spread(key.hashCode());int binCount = 0;for (Node<K,V>[] tab = table;;) {Node<K,V> f; int n, i, fh;//若为空则初始化,ConcurrentHash new时不会初始化容器if (tab == null || (n = tab.length) == 0)tab = initTable();//若发现数组索引位下无元素则采用cas乐观锁的方式进行put。else if ((f = tabAt(tab, i = (n - 1) & hash)) == null) {if (casTabAt(tab, i, null,new Node<K,V>(hash, key, value, null)))break;                   // no lock when adding to empty bin}//若上述条件不满住且当前槽位的hash值为-1则说明当前正在扩容中并加速扩容,等待下一次循环走入下方的elseelse if ((fh = f.hash) == MOVED)tab = helpTransfer(tab, f);else {V oldVal = null;//若当前数组索引下的元素不为空则说明出现hash冲突,可以用链表存储则将当前元素加锁synchronized (f) {if (tabAt(tab, i) == f) {if (fh >= 0) {binCount = 1;for (Node<K,V> e = f;; ++binCount) {K ek;//判断新key与遍历的元素key是否一致,若一致则直接覆盖if (e.hash == hash &&((ek = e.key) == key ||(ek != null && key.equals(ek)))) {oldVal = e.val;if (!onlyIfAbsent)e.val = value;break;}Node<K,V> pred = e;//若发现链表中没有一致的key,那就就将新元素插入到最后一个元素的后面if ((e = e.next) == null) {pred.next = new Node<K,V>(hash, key,value, null);break;}}}//若发现“链表”为树的结构则通过红黑树的方式进行putelse if (f instanceof TreeBin) {Node<K,V> p;binCount = 2;if ((p = ((TreeBin<K,V>)f).putTreeVal(hash, key,value)) != null) {oldVal = p.val;if (!onlyIfAbsent)p.val = value;}}}}//最终判判断决定是否将链表结构是否要转换成红黑树if (binCount != 0) {if (binCount >= TREEIFY_THRESHOLD)treeifyBin(tab, i);if (oldVal != null)return oldVal;break;}}}addCount(1L, binCount);return null;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/319773.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

鸿蒙系列--装饰器

一、基础UI组件结构 每个UI组件需要定义为Component struct对象&#xff0c;其内部必须包含一个且只能包含一个build(){}函数&#xff0c;用于绘制UI&#xff1b;struct之内、build()函数之外的地方用于存放数据。 二、基本UI装饰器 Entry 装饰struct&#xff0c;页面的入口…

.cer格式证书文件和 .pfx格式证书文件有什么区别?

这里我们将讨论.cer和.pfx文件类型之间的差异。 什么是数字证书&#xff1f; 数字证书在电子通信中用作验证身份的密码机制。我们需要这些证书来建立安全的在线通信渠道&#xff0c;并确保数字数据的隐私、真实性和正确性。 数字证书包括主题&#xff08;实体详细信息&#xf…

MyBatis学习二:Mapper代理开发、配置文件完成增删改查、注解开发

前言 公司要求没办法&#xff0c;前端也要了解一下后端知识&#xff0c;这里记录一下自己的学习 学习教程&#xff1a;黑马mybatis教程全套视频教程&#xff0c;2天Mybatis框架从入门到精通 文档&#xff1a; https://mybatis.net.cn/index.html Mapper代理开发 目的 解决…

sql:定时执行存储过程(嵌套存储过程、使用游标)

BEGINDeclare FormNo nvarchar(20) --单号Declare Type nvarchar(50) --类型Declare PickedQty float -Declare OutQty float Declare 生产量 floatDeclare 已装箱数量 float Declare 已入库数量 floatDeclare 损耗数量 float Declare 退货品出库数量 intdeclare k c…

C#:如何产生一个临时文件

在我们实际编程中&#xff0c;经常有将内容写到一个临时文件的需要。 比如&#xff1a;将网络上的图片下载下来&#xff0c;获取到图片的一些信息。 代码如下&#xff0c;看结果可知&#xff1a; 临时文件都是保存在系统临时文件夹的目录下&#xff0c;临时文件的扩展名统一…

程序媛的mac修炼手册-- 终端shell的驾驭 zsh vs bash

进入终端(Terminal)为新下载的应用配置环境&#xff0c;是Mac生产力up up的关键一步&#xff0c;更是编程小白装大神的第一步。Fake it till you make it , 硅谷大神标准路径&#xff5e; shell的基本原理 为应用配置环境&#xff0c;相当于在应用和操作系统间架桥。由此&…

手机视频监控客户端APP如何实现跨安卓、苹果和windows平台,并满足不同人的使用习惯

目 录 一、手机视频监控客户端的应用和发展 二、手机视频监控客户端存在的问题 三、HTML5视频监控客户端在手机上实现的方案 &#xff08;一&#xff09;HTML5及其优点 &#xff08;二&#xff09;HTML5在手机上实现视频应用功能的优势 四、手机HTML5…

Day22 二叉树part08 235.二叉搜索树的最近公共祖先 701.二叉搜索树中的插入操作 450.删除二叉搜索树中的节点

二叉树part08 235.二叉搜索树的最近公共祖先 701.二叉搜索树中的插入操作 450.删除二叉搜索树中的节点 235. 二叉搜索树的最近公共祖先 方法一&#xff1a;递归法&#xff08;利用二叉搜索树性质&#xff09; class Solution { public:TreeNode* lowestCommonAncestor(TreeN…

【C语言】编程世界的不朽基石与未来展望

C语言&#xff0c;一种经久不衰的高级编程语言&#xff0c;自1972年由Dennis Ritchie在AT&T贝尔实验室开发以来&#xff0c;已深深扎根于编程语言的发展历程中。它既是计算机科学史上的一个重要里程碑&#xff0c;也是现代软件开发的核心支柱。从操作系统到嵌入式系统的构建…

[C#]C# OpenVINO部署yolov8实例分割模型

【官方框架地址】 https://github.com/ultralytics/ultralytics.git 【算法介绍】 YOLOv8 抛弃了前几代模型的 Anchor-Base。 YOLO 是一种基于图像全局信息进行预测的目标检测系统。自 2015 年 Joseph Redmon、Ali Farhadi 等人提出初代模型以来&#xff0c;领域内的研究者们…

【Adobe Acrobat DC】弹窗:正在准备文档以供阅读,请稍候

问题描述 Acrobat打开PDF就会有这个弹窗&#xff0c;要所有扫一遍才能看&#xff0c;要等特别久 解决办法1 参考这篇博客评论区里给出的方法 1. 转到“编辑”>“首选项”>“朗读”。 2. 在“屏幕阅读器选项”下面&#xff0c;选择“仅阅读当前可见页面”。 再次打开…

程序员必知!享元模式的实战应用与案例分析

享元模式是一种减少相似对象创建和销毁的设计模式&#xff0c;通过将对象状态分为不变和可变部分&#xff0c;实现内存节省和性能提升。例如&#xff0c;在线游戏中大量玩家角色可共享相同的不变属性&#xff0c;而每人特有的可变属性则单独存储&#xff0c;享元模式使用享元类…