深入理解Java中的线程安全List:CopyOnWriteArrayList原理和应用

在这里插入图片描述

码到三十五 : 个人主页

心中有诗画,指尖舞代码,目光览世界,步履越千山,人间尽值得 !


在Java并发编程中,线程安全的数据结构是至关重要的。其中,CopyOnWriteArrayList是一个线程安全的ArrayList实现,它提供了在并发环境下对列表进行读写的功能。本文将深入探讨CopyOnWriteArrayList的工作原理、使用场景以及潜在的性能问题。

目录

    • 1️⃣ 什么是CopyOnWrite(写时复制)
    • 2️⃣什么是CopyOnWriteArrayList
    • 3️⃣CopyOnWriteArrayList的工作原理
      • 3.1 读写分离的设计模式的几个优点
      • 3.2 存在的性能问题
    • 3️⃣CopyOnWriteArrayList使用场景
    • 4️⃣CopyOnWriteArrayList的应用
    • 5️⃣总结

1️⃣ 什么是CopyOnWrite(写时复制)

CopyOnWrite,也被称为写时复制(Copy-On-Write,简称COW),是程序设计领域中的一种优化策略。这种策略的核心思想是,当多个调用者(或线程)同时访问同一份资源时,他们会共同获取一个指向该资源的指针。只要没有调用者尝试修改这份资源,所有的调用者都可以继续访问同一个资源。但是,一旦有调用者尝试修改资源,系统就会复制一份该资源的副本给这个调用者,而其他调用者所见到的仍然是原来的资源。这个过程对其他的调用者都是透明的,他们并不知道资源已经被复制。

在Java中,CopyOnWriteArrayList和CopyOnWriteArraySet就是使用了这种策略的两个类。这两个类都位于java.util.concurrent包下,是线程安全的集合类。当需要修改集合中的元素时,它们不会直接在原集合上进行修改,而是复制一份新的集合,然后在新的集合上进行修改。修改完成后,再将指向原集合的引用指向新的集合。这种设计使得读操作可以在不加锁的情况下进行,从而提高了并发性能。

总的来说,CopyOnWrite是一种适用于读多写少场景的优化策略,它通过复制数据的方式实现了读写分离,提高了并发性能。但是,它也存在一些潜在的性能问题,如内存占用增加、写操作性能下降以及频繁的垃圾回收。因此,在使用时需要根据具体场景进行权衡和选择。

2️⃣什么是CopyOnWriteArrayList

CopyOnWriteArrayList是Java并发包java.util.concurrent中的一个类,它实现了List接口。如其名所示,

CopyOnWriteArrayList是Java中的一个类,位于java.util.concurrent包下。它是ArrayList的一个线程安全的变体,其中所有可变操作(如add和set等)都是通过创建底层数组的新副本来实现的,因此被称为“写时复制”的列表。

由于CopyOnWriteArrayList在遍历时不会对列表进行任何修改,因此它绝对不会抛出ConcurrentModificationException的异常。它在修改操作(如addset等)时,会复制一份底层数组,然后在新的数组上进行修改,修改完成后再将指向底层数组的引用切换到新的数组。这种设计使得读操作可以在不加锁的情况下进行,从而提高了并发性能,这个特性使得它在多线程环境下进行遍历操作时更为安全。

然而,CopyOnWriteArrayList并没有“扩容”的概念。每次写操作(如add或remove)都需要复制一个全新的数组,这在写操作较为频繁时可能会导致性能问题,因为复制整个数组的操作是相当耗时的。因此,在使用CopyOnWriteArrayList时,需要特别注意其适用场景,一般来说,它更适合于读多写少的场景。

3️⃣CopyOnWriteArrayList的工作原理

CopyOnWriteArrayList是ArrayList的一个线程安全的变体。读操作可以在不加锁的情况下进行,从而提高了并发性能。

具体来说,CopyOnWriteArrayList内部有一个可重入锁(ReentrantLock)来保证线程安全,但这个锁只在写操作时才会被使用。当进行修改操作时,线程会先获取锁,然后复制底层数组,并在新数组上执行修改。修改完成后,通过volatile关键字修饰的引用来确保新的数组对所有线程可见。由于读操作不需要获取锁,因此多个线程可以同时进行读操作,而不会相互干扰。
在这里插入图片描述

3.1 读写分离的设计模式的几个优点

  • 读操作性能很高

由于读操作不需要获取锁,因此多个线程可以同时进行读操作,而不会相互干扰。这使得在高并发场景下,CopyOnWriteArrayList的读操作性能非常出色。

  • 数据一致性

由于写操作是通过复制底层数组并在新数组上执行修改来实现的,因此不会出现多个线程同时修改同一个元素的情况。这保证了数据的一致性。

  • 适用于读多写少的场景

由于写操作需要复制整个底层数组,因此在写操作较为频繁的场景下,CopyOnWriteArrayList的性能可能会受到较大影响。但在读多写少的场景下,它可以充分发挥其优势。

3.2 存在的性能问题

  • 内存占用

每次写操作都需要复制整个底层数组,这会导致内存占用增加。特别是在列表较大时,这种内存开销可能会变得非常显著。

  • 写操作性能下降

由于每次写操作都需要复制整个数组,并在新数组上执行修改,因此写操作的性能可能会受到较大影响。特别是在高并发场景下,这种性能下降可能会更加明显。

  • 频繁的垃圾回收

由于写操作会创建新的数组,因此可能导致频繁的垃圾回收。这可能会对系统的整体性能产生影响。

总的来说,CopyOnWriteArrayList是一种适用于读多写少场景的线程安全列表实现。它通过复制底层数组的方式实现了读写分离,提高了读操作的并发性能。但在使用时需要根据具体场景进行权衡和选择,以避免潜在的性能问题。

3️⃣CopyOnWriteArrayList使用场景

CopyOnWriteArrayList适用于读多写少的场景。在这种场景下,由于读操作不需要获取锁,因此可以充分发挥多核CPU的并行计算能力,提高系统的吞吐量。然而,在写操作较为频繁的场景下,CopyOnWriteArrayList的性能可能会受到较大影响。

4️⃣CopyOnWriteArrayList的应用

下面是一个使用CopyOnWriteArrayList的代码,它模拟了一个简单的新闻发布系统。在这个系统中,多个线程可以并发地添加新闻和读取新闻列表。由于读操作远多于写操作,因此使用CopyOnWriteArrayList是合适的。

import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;// 新闻类
class News {private String title;private String content;public News(String title, String content) {this.title = title;this.content = content;}public String getTitle() {return title;}public String getContent() {return content;}@Overridepublic String toString() {return "News{" +"title='" + title + '\'' +", content='" + content + '\'' +'}';}
}// 新闻发布系统类
public class NewsPublisherSystem {// 使用CopyOnWriteArrayList存储新闻列表private final List<News> newsList = new CopyOnWriteArrayList<>();// 添加新闻public void addNews(News news) {newsList.add(news);System.out.println("新闻已添加: " + news);}// 获取新闻列表public List<News> getNewsList() {return newsList;}// 模拟多线程添加和读取新闻public void simulate() {ExecutorService executor = Executors.newFixedThreadPool(10);// 提交5个添加新闻的任务for (int i = 0; i < 5; i++) {final int index = i;executor.submit(() -> {for (int j = 0; j < 10; j++) {News news = new News("新闻标题" + index + "-" + j, "新闻内容" + index + "-" + j);addNews(news);try {// 模拟新闻发布的延迟TimeUnit.MILLISECONDS.sleep(50);} catch (InterruptedException e) {e.printStackTrace();}}});}// 提交5个读取新闻列表的任务for (int i = 0; i < 5; i++) {executor.submit(() -> {for (int j = 0; j < 20; j++) {System.out.println("当前新闻列表: " + getNewsList());try {// 模拟读取新闻列表的延迟TimeUnit.MILLISECONDS.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}});}// 关闭执行器服务executor.shutdown();try {if (!executor.awaitTermination(1, TimeUnit.MINUTES)) {executor.shutdownNow();}} catch (InterruptedException e) {executor.shutdownNow();Thread.currentThread().interrupt();}}public static void main(String[] args) {NewsPublisherSystem system = new NewsPublisherSystem();system.simulate();}
}

NewsPublisherSystem类维护了一个CopyOnWriteArrayList来存储新闻对象。addNews方法用于添加新闻到列表中,而getNewsList方法用于获取当前的新闻列表。

simulate方法中,我们创建了一个固定大小的线程池,并提交了10个任务:其中5个任务用于添加新闻,另外5个任务用于读取新闻列表。每个添加新闻的任务会创建并添加10条新闻,而每个读取新闻列表的任务会读取新闻列表20次。

由于使用了CopyOnWriteArrayList,多个线程可以同时读取新闻列表,而不会有线程安全问题。当添加新闻时,CopyOnWriteArrayList会复制底层数组,从而保证读取操作不会受到写操作的影响。

请注意,由于CopyOnWriteArrayList在写操作时会复制整个底层数组,因此在新闻列表非常大且写操作频繁的情况下,性能可能会受到影响。在这种情况下,可能需要考虑其他并发数据结构或同步策略。然而,在本案例中,由于读操作远多于写操作,使用CopyOnWriteArrayList是合适的。

5️⃣总结

CopyOnWriteArrayList是Java并发编程中一个重要的线程安全列表实现。它通过复制底层数组的方式实现了读写分离,提高了读操作的并发性能。然而,它也存在一些潜在的性能问题,如内存占用增加、写操作性能下降以及频繁的垃圾回收。因此,在使用CopyOnWriteArrayList时,需要根据具体的使用场景进行权衡和选择。在读多写少的场景下,CopyOnWriteArrayList可以发挥出色的性能;而在写操作较为频繁的场景下,可能需要考虑其他线程安全的列表实现。

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

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

相关文章

HBase非关系型数据库

HBase非关系型数据库 1 什么是HBase2 HBase的特点3 什么时候需要HBase4 HBase的数据模型5 HBase架构5.1 架构5.2 HBase如何列式储存 6 如何正确设计RowKey 1 什么是HBase HBase – Hadoop Database&#xff0c;是一个高可靠性、高性能、面向列、可伸缩、 实时读写的分布式数据…

第100+1步 ChatGPT文献复现:ARIMAX预测肺结核 vol. 1

基于WIN10的64位系统演示 一、写在前面 各位大佬&#xff0c;好久不见。 《100步入门机器学习》肝完了&#xff0c;不懂大家学了多少了&#xff0c;默认你们都学完了吧。 今年我们换一个玩法&#xff08;灌水&#xff09;&#xff1a;一系列更接近实战的教程&#xff0c;复…

(黑马出品_07)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式

&#xff08;黑马出品_07&#xff09;SpringCloudRabbitMQDockerRedis搜索分布式 微服务技术分布式搜索 今日目标1.数据聚合1.1.聚合的种类1.2.DSL实现聚合1.2.1.Bucket聚合语法1.2.2.聚合结果排序1.2.3.限定聚合范围1.2.4.Metric聚合语法1.2.5.小…

6.S081的Lab学习——Lab1: Xv6 and Unix utilities

文章目录 前言一、启动xv6(难度&#xff1a;Easy)解析&#xff1a; 二、sleep(难度&#xff1a;Easy)解析&#xff1a; 三、pingpong&#xff08;难度&#xff1a;Easy&#xff09;解析&#xff1a; 四、Primes(素数&#xff0c;难度&#xff1a;Moderate/Hard)解析&#xff1a…

node.js入门

一、cmd常用命令 windowsR 再输入cmd 打开命令提示符 (1)切换盘符 c: / d: (2)dir 查看全部内容 (3)cd 你需要打开的文件所处的大文件夹名字 (4)cd 大文件夹名再输入dir 查看该大文件里面的所有内容 (5)输出结果里 一个.表示当前目录&#xff0c;两个..表示上级目录 …

怎样在CSDN赚点零花钱

请教一下各位大佬&#xff0c;看到你们在CSDN很多都几万粉丝以上&#xff0c;能不能分享一下有什么涨粉的经验&#xff0c;还有怎样转化为额外收益……感谢各位提供宝贵的经验&#xff0c;谢谢……

文献阅读笔记:全卷积神经网络

文献阅读笔记&#xff1a;全卷积神经网络 摘要Abstract1. 全卷积神经网络1.1 文献摘要1.2 全卷积神经网络1.2.1 网络结构1.2.0 从分类器到密集 FCN1.2.2 上采样 Upsampling1.2.3 跳级结构1.2.4 FCN训练 1.3 实验1.4 总结 2. 代码实现 摘要 本周学习了全卷积神经网络&#xff0…

Acer宏碁非凡Swift SFG16-71工厂模式原厂Win11系统,预装OEM系统恢复开箱状态

宏基笔记本电脑SFG16-71原装出厂Windows11系统安装工厂包下载&#xff0c;带恢复重置功能 链接&#xff1a;https://pan.baidu.com/s/1JK02kBbwKG_cIBNlEOzrOw?pwdzdfm 提取码&#xff1a;zdfm 原装工厂包系统自带所有驱动、Office办公软件、出厂时自带主题壁纸图片、系统…

如何免费获取基于公网 IP 的 SSL 证书 (无需域名)

现在给网站安装SSL证书来实现网站的HTTPS安全访问已经成了大多数人的共识&#xff0c;但是有一些特殊情况&#xff1a;比如对于个别的应用IP地址不需要绑定域名&#xff0c;只是单纯用IP来访问网站&#xff0c;这种情况下&#xff0c;可以实现HTTPS访问吗&#xff1f; 先说答案…

vue-创建vue项目记录

安装node.js 先安装node.js的运行环境node.js的下载地址 安装后就可以使用npm命令 1、清除npm缓存&#xff1a;npm cache clean --force 2、禁用SSL&#xff1a;npm config set strict-ssl false 3、手动设置npm镜像源&#xff1a;npm config set registry https://registry.…

揭密无文件勒索病毒攻击,思考网络安全新威胁

前言 最近几年基于无文件攻击的网络犯罪活动越来越多&#xff0c;一些网络犯罪团伙开发了各种基于无文件攻击的恶意软件攻击套件&#xff0c;这些恶意软件攻击套件可用于勒索病毒、挖矿病毒、RAT远控、僵尸网络等恶意软件&#xff0c;在过去的几年时间里&#xff0c;无文件感染…

L1-8 静静的推荐(Python)

天梯赛结束后&#xff0c;某企业的人力资源部希望组委会能推荐一批优秀的学生&#xff0c;这个整理推荐名单的任务就由静静姐负责。企业接受推荐的流程是这样的&#xff1a; 只考虑得分不低于 175 分的学生&#xff1b;一共接受 K 批次的推荐名单&#xff1b;同一批推荐名单上…

Day35:安全开发-JavaEE应用原生反序列化重写方法链条分析触发类类加载

目录 Java-原生使用-序列化&反序列化 Java-安全问题-重写方法&触发方法 Java-安全问题-可控其他类重写方法 思维导图 Java知识点&#xff1a; 功能&#xff1a;数据库操作&#xff0c;文件操作&#xff0c;序列化数据&#xff0c;身份验证&#xff0c;框架开发&…

pandas plot函数:数据可视化的快捷通道

一般来说&#xff0c;我们先用pandas分析数据&#xff0c;然后用matplotlib之类的可视化库来显示分析结果。而pandas库中有一个强大的工具--plot函数&#xff0c;可以使数据可视化变得简单而高效。 1. plot 函数简介 plot函数是pandas中用于数据可视化的一个重要工具&#xff0…

凌鲨本地接口架构

本地API通过监听本地端口&#xff0c;提供http服务&#xff0c;让本地应用可以获取信息和操作凌鲨客户端。 本地API架构 #mermaid-svg-seodZa6VsI4Qc8Cj {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-seodZa6VsI4…

辽宁博学优晨教育:视频剪辑培训的安全正规之路

在当今数字化时代&#xff0c;视频剪辑已成为一项炙手可热的技能。为满足广大学习者的需求&#xff0c;辽宁博学优晨教育推出了一系列专业的视频剪辑培训课程。本文将重点介绍辽宁博学优晨教育的视频剪辑培训如何在保障学员安全和学习效果方面做出了卓越的努力。 一、正规资质&…

Linux操作系统-06-进程与服务管理

使用ps命令查看进程。包括过滤进程信息 使用systemctl命令管理和运行Linux服务 进程&#xff08;Process&#xff09;&#xff1a;操作系统正在运行的应用程序。任意一个进程&#xff0c;都会消耗CPU和内存资源&#xff0c; 服务&#xff08;Service&#xff09;&#xff1a…

Word中解决插入脚注导致的分页位置错误问题

先放一个截图&#xff1a; 上面的截图中&#xff0c;样式为标题3的段落“四、固执的念头”前插入了连续型分节符&#xff0c;并且该分节符的样式为正文&#xff0c;前后的正文段落中有脚注&#xff0c;结果在分页时&#xff0c;标题3段落“四、固执的念头”后的正文段落自动进入…

【JAVA】我和“JAVA“的细水长流,输入输出

&#x1f308;个人主页: Aileen_0v0 &#x1f525;热门专栏: 华为鸿蒙系统学习|计算机网络|数据结构与算法|MySQL| ​&#x1f4ab;个人格言:“没有罗马,那就自己创造罗马~” #mermaid-svg-Ck59fYFNNdpuGmVT {font-family:"trebuchet ms",verdana,arial,sans-serif;f…

JVM-3

HotSpot虚拟机对象 我在网上看了很多相关的文章&#xff0c;发现在创建对象和对象的结构中内容都不太一样&#xff0c;一些关键字也很不同&#xff0c;于是我通过参考《深入理解Java虚拟机》这本书&#xff0c;自己总结了一篇。 1.对象的创建 当JVM收到一条创建对象的字节码…