Java多线程八股(三)一>多线程环境使用哈希表和ArrayList

目录:

一.多线程环境使用ArrayList:

二.多线程使用哈希表:

一.多线程环境使用ArrayList:

首先我们知道,Vector, Stack, HashTable, 是线程安全的(但是不建议用), 其他的集合类不是线程安全的 ,下面是 ArrayList的使用场景
1.自行对代码加锁 (推荐使用)


2. 套个壳使用Collections.synchronizedList(new ArrayList)

就类似于上面的HashTable这些,但是不推荐使用因为使用加锁是有代价的(性能开销等等)
synchronizedList 是标准库提供的一个基于 synchronized 进行线程同步的 List. synchronizedList的关键操作方法上都带有 synchronized


3.使用 CopyOnWriteArrayList容器:

这个容器不是加锁设计的,而是用编程常见的一种思想方法->写时拷贝

以下是 写时拷贝的讲解:
注意:CopyOnWriteArrayList容器也不是所有场景都适合,当数组非常大和涉及多个线程修改时就会可能出错



二.多线程使用哈希表:
HashMap 本身不是线程安全的
在多线程环境下使用哈希表可以使用: Hashtable ConcurrentHashMap


1.Hashtable
只是简单的把关键方法加上了 synchronized 关键字,(相当给this加锁,就是给整个哈希表加锁)
这样如果修改两个元素 不在同一个链表上,就是修改不同元素 不会发生线程安全问题
但是如果在同一个链表上就可能会发生线程安全问题 (比如把两个元素插入到第一个元素后面)



2.ConcurrentHashMap:
(1). 读操作没有加锁(但是使用了 volatile 保证从内存读取结果)


(2). 只对写操作进行加锁. 加锁的方式仍然是是⽤ synchronized, 但是不是锁整个对象, 而是"锁桶" (用每个链表的头结点作为锁对象), 大大降低了锁冲突的概率.


(3).充分利用CAS(原子类)特性. 比如 size 属性通过 CAS 来更新. 避免出现重量级锁的情况也避免了线程安全问题. (通过原子类修饰size)


(4).优化了扩容方式: 化整为零
化整为零解释:
扩容需要把旧的哈希表元素全部搬运一次性搬运到新的哈希表,这个时候如果元素太多搬运完,势必进行更长时间的加锁解锁,会很耗时间,这个时候就可以采用"化整为零"的方式,不是一次搬运完,而是多次进行搬运(一次搬运百分之几),后续再进行基本操作,这样加锁解锁耗时就很短,分多次部分触发完成全部任务

相关推荐
康小庄10 小时前
通过NGINX实现将小程序HTTPS请求转为内部HTTP请求
java·spring boot·nginx·spring·http·小程序
Swift社区10 小时前
Date / LocalDateTime 转换错误,一次踩坑后的完整复盘
java·spring boot·spring
知乎的哥廷根数学学派10 小时前
基于物理约束指数退化与Hertz接触理论的滚动轴承智能退化趋势分析(Pytorch)
开发语言·人工智能·pytorch·python·深度学习·算法·机器学习
长路归期无望10 小时前
一步步入门机器人【Arduino基础】
开发语言·经验分享·笔记·学习·机器人
源文雨10 小时前
批量递归转换 mp4 为 flac/m4a 的 bash 脚本
开发语言·ffmpeg·bash·转码·mp4·m4a·flac
Seven9710 小时前
PriorityQueue的秘密:堆结构的高效应用与实现原理
java
不绝19110 小时前
C#进阶:协程与事件
开发语言·c#
feifeigo12310 小时前
斜激波参数计算MATLAB程序
开发语言·matlab
小小前端--可笑可笑10 小时前
【Three.js + MediaPipe】视频粒子特效:实时运动检测与人物分割技术详解
开发语言·前端·javascript·音视频·粒子特效
奔跑的web.10 小时前
JavaScript 对象属性遍历Object.entries Object.keys:6 种常用方法详解与对比
开发语言·前端·javascript·vue.js