CopyOnWriteArrayList介绍和使用

2024-03-12 21:44

本文主要是介绍CopyOnWriteArrayList介绍和使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. CopyOnWriteArrayList是什么

CopyOnWriteArrayList 是 Java.util.concurrent 包中的一个线程安全的 List 实现类。它通过在修改操作时创建底层数组的副本来实现线程安全,从而保证了并发访问的一致性。它适用于读操作频繁、写操作较少的场景。

2. CopyOnWriteArrayList特性

  1. 线程安全性:
    CopyOnWriteArrayList 是线程安全的,可以在多线程环境下并发访问而无需额外的同步措施。它通过在修改操作(如添加、修改、删除元素)时创建底层数组的副本,并在副本上进行修改操作,以确保原始数组的线程安全性。这样,读取操作可以在不阻塞的情况下同时进行,不会影响其他线程的读取操作。
  2. 内部实现:
    CopyOnWriteArrayList 内部使用一个可变数组来存储元素。在进行修改操作时,会创建一个新的数组副本,并将修改操作应用于副本,然后将副本替换原始数组。这种写时复制的机制保证了修改操作的线程安全性,但也带来了一些额外的开销和内存消耗。
  3. 迭代器一致性:
    CopyOnWriteArrayList 提供的迭代器是弱一致性的。即,在迭代过程中,如果有其他线程对列表进行了修改,迭代器不会抛出 ConcurrentModificationException 异常,而是仍然基于迭代器创建时的快照进行迭代。这意味着迭代器不会遇到并发修改的异常,但可能会遇到迭代过程中数据的变化。
  4. 适用场景:
    CopyOnWriteArrayList 适用于读操作频繁、写操作较少的场景。由于每次写操作都会创建底层数组的副本,因此写操作的开销较大,不适合频繁的修改操作。但对于读操作,由于读取操作不需要额外的同步措施,所以可以在并发访问下获得较好的性能。

需要注意的是,由于 CopyOnWriteArrayList 每次修改操作都会创建副本,因此它的内存消耗较大。因此,对于存储大量数据的情况,应谨慎使用,以免导致内存占用过高。

CopyOnWriteArrayList 提供了一种线程安全的 List 实现,适用于读操作频繁、写操作较少的场景,具有良好的并发性能和一致性保证。

3. CopyOnWriteArrayList 的线程安全原理

CopyOnWriteArrayList 的线程安全原理是通过写时复制(Copy-On-Write)的机制实现的。

当进行修改操作(如添加、修改、删除元素)时,CopyOnWriteArrayList 会创建一个底层数组的副本,并在副本上进行修改操作,而不是直接在原始数组上进行修改。这样做的目的是保证并发访问的线程安全性。

  1. 在读取操作时,多个线程可以同时进行读取,而不需要额外的同步措施。因为读取操作直接读取当前的数组,不会受到其他线程修改操作的影响。
  2. 在写操作时,CopyOnWriteArrayList 会创建一个新的数组副本,并在副本上进行修改操作。这样做的好处是,写操作不会影响其他线程的读取操作。
  3. 当修改操作完成后,CopyOnWriteArrayList 会将新的数组副本替换原始数组。这个替换操作是原子的,确保了新的数组副本对其他线程的可见性。

通过使用写时复制的机制,CopyOnWriteArrayList 实现了线程安全。每个线程在进行修改操作时都会操作自己的副本,不会对其他线程的读取操作造成影响。这样就避免了传统的同步机制(如锁)带来的竞争和阻塞,提高了并发性能。

4. 源码解析

4.1. add方法

public class CopyOnWriteArrayList<E>
implements List<E>, RandomAccess, Cloneable, java.io.Serializable {private static final long serialVersionUID = 8673264195747942595L;/** The lock protecting all mutators */// 可重入锁final transient ReentrantLock lock = new ReentrantLock();/** The array, accessed only via getArray/setArray. */// volatile 关键字修饰private transient volatile Object[] array;/*** Gets the array.  Non-private so as to also be accessible* from CopyOnWriteArraySet class.*/// 获取数组final Object[] getArray() {return array;}/*** Sets the array.*/// 设置数组final void setArray(Object[] a) {array = a;}/*** Appends the specified element to the end of this list.** @param e element to be appended to this list* @return {@code true} (as specified by {@link Collection#add})*/public boolean add(E e) {final ReentrantLock lock = this.lock;// 加锁lock.lock();try {// 获取原数组Object[] elements = getArray();// 原数组长度int len = elements.length;// 拷贝原数组到新数组,且新数组长度=未原数组长度+1,因为要新增一个元素Object[] newElements = Arrays.copyOf(elements, len + 1);// 新增元素放在数组最后一位newElements[len] = e;// 新数组替换掉原来的数组setArray(newElements);return true;} finally {// 解锁lock.unlock();}}/*** Inserts the specified element at the specified position in this* list. Shifts the element currently at that position (if any) and* any subsequent elements to the right (adds one to their indices).** @throws IndexOutOfBoundsException {@inheritDoc}*/public void add(int index, E element) {final ReentrantLock lock = this.lock;// 加锁lock.lock();try {// 获取原数组Object[] elements = getArray();// 数组长度int len = elements.length;// 插入的位置大于原数组长度或者小于0就报错if (index > len || index < 0)throw new IndexOutOfBoundsException("Index: "+index+// 新建数组                                        ", Size: "+len);Object[] newElements;// 原数组长度 - 插入位置索引int numMoved = len - index;// 如果原数组长度 - 插入位置索引 = 0if (numMoved == 0)// 表示插入的是在最尾部,新数组等于元数组,然后长度加一newElements = Arrays.copyOf(elements, len + 1);else {// 否则的话表示 插入在数组中间位置,就需要两次拷贝数组// 新数组长度等于元数组加一newElements = new Object[len + 1];// 第一次拷贝数组 从0到index。System.arraycopy(elements, 0, newElements, 0, index);// 第二次拷贝数组 从index+1到末尾。System.arraycopy(elements, index, newElements, index + 1,numMoved);}// 给index索引位置赋值newElements[index] = element;// 新数组替换掉原来的数组 setArray(newElements);} finally {// 解锁lock.unlock();}}
}

4.2. System.arraycopy()介绍

System.arraycopy() 是Java中的一个方法,用于将一个数组中的元素复制到另一个数组中。它的语法如下:

System.arraycopy(Object src, int srcPos, Object dest, int destPos, int length)

参数说明:

  • src:源数组,即要复制的元素来源。
  • srcPos:源数组中要复制的起始位置。
  • dest:目标数组,即要将元素复制到的数组。
  • destPos:目标数组中开始存放复制元素的起始位置。
  • length:要复制的元素个数。

所以,System.arraycopy(elements, 0, newElements, 0, index) 的意思是将 elements 数组中从索引 0 开始的 index 个元素复制到 newElements 数组中,复制的起始位置也从目标数组的索引 0 开始。这个方法可以用来执行数组的部分复制或者数组的重排操作。

具体例子:

public static void main(String[] args) {Object[] elements = {1, 2, 3, 4, 5};int index = 3;int len = elements.length;int numMoved = len - index;Object[] newElements = new Object[len + 1];System.arraycopy(elements, 0, newElements, 0, index);System.out.println("第1次拷贝结果:" + Arrays.toString(newElements));System.arraycopy(elements, index, newElements, index + 1, numMoved);System.out.println("第2次拷贝结果:" + Arrays.toString(newElements));
}

控制台输出结果

img

4.3. remove方法

/*** Removes the element at the specified position in this list.* Shifts any subsequent elements to the left (subtracts one from their* indices).  Returns the element that was removed from the list.** @throws IndexOutOfBoundsException {@inheritDoc}*/
public E remove(int index) {final ReentrantLock lock = this.lock;// 加锁lock.lock();try {// 获取原数组Object[] elements = getArray();// 原数组长度int len = elements.length;// 先得index位置的值E oldValue = get(elements, index);// 原数组长度 - index - 1int numMoved = len - index - 1;// 如果要删除的数据正好是数组的尾部,直接删除if (numMoved == 0)// 复制原数组,数组长度少1setArray(Arrays.copyOf(elements, len - 1));else {// 如果删除的数据在数组的中间,分三步走// 1. 设置新数组的长度减一,因为是减少一个元素// 2. 从 0 拷贝到数组新位置// 3. 从新位置拷贝到数组尾部Object[] newElements = new Object[len - 1];System.arraycopy(elements, 0, newElements, 0, index);System.arraycopy(elements, index + 1, newElements, index,numMoved);// 新数组替换掉原来的数组 setArray(newElements);}return oldValue;} finally {// 解锁lock.unlock();}
}

4.4. 总结

通过分析源码可以看出,添加、删除都是先复制原数组到新数组,然后再新数组中操作新增、删除元素,最后再替换原来的数组,再解锁。

步骤如下:

  1. 加锁
  2. 从原数组中拷贝出新数组
  3. 在新数组上进行操作,并把新数组赋值给数组容器
  4. 解锁

这篇关于CopyOnWriteArrayList介绍和使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/802667

相关文章

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

使用Python实现可恢复式多线程下载器

《使用Python实现可恢复式多线程下载器》在数字时代,大文件下载已成为日常操作,本文将手把手教你用Python打造专业级下载器,实现断点续传,多线程加速,速度限制等功能,感兴趣的小伙伴可以了解下... 目录一、智能续传:从崩溃边缘抢救进度二、多线程加速:榨干网络带宽三、速度控制:做网络的好邻居四、终端交互

Python中注释使用方法举例详解

《Python中注释使用方法举例详解》在Python编程语言中注释是必不可少的一部分,它有助于提高代码的可读性和维护性,:本文主要介绍Python中注释使用方法的相关资料,需要的朋友可以参考下... 目录一、前言二、什么是注释?示例:三、单行注释语法:以 China编程# 开头,后面的内容为注释内容示例:示例:四

Python中win32包的安装及常见用途介绍

《Python中win32包的安装及常见用途介绍》在Windows环境下,PythonWin32模块通常随Python安装包一起安装,:本文主要介绍Python中win32包的安装及常见用途的相关... 目录前言主要组件安装方法常见用途1. 操作Windows注册表2. 操作Windows服务3. 窗口操作

Go语言数据库编程GORM 的基本使用详解

《Go语言数据库编程GORM的基本使用详解》GORM是Go语言流行的ORM框架,封装database/sql,支持自动迁移、关联、事务等,提供CRUD、条件查询、钩子函数、日志等功能,简化数据库操作... 目录一、安装与初始化1. 安装 GORM 及数据库驱动2. 建立数据库连接二、定义模型结构体三、自动迁

ModelMapper基本使用和常见场景示例详解

《ModelMapper基本使用和常见场景示例详解》ModelMapper是Java对象映射库,支持自动映射、自定义规则、集合转换及高级配置(如匹配策略、转换器),可集成SpringBoot,减少样板... 目录1. 添加依赖2. 基本用法示例:简单对象映射3. 自定义映射规则4. 集合映射5. 高级配置匹

Spring 框架之Springfox使用详解

《Spring框架之Springfox使用详解》Springfox是Spring框架的API文档工具,集成Swagger规范,自动生成文档并支持多语言/版本,模块化设计便于扩展,但存在版本兼容性、性... 目录核心功能工作原理模块化设计使用示例注意事项优缺点优点缺点总结适用场景建议总结Springfox 是

嵌入式数据库SQLite 3配置使用讲解

《嵌入式数据库SQLite3配置使用讲解》本文强调嵌入式项目中SQLite3数据库的重要性,因其零配置、轻量级、跨平台及事务处理特性,可保障数据溯源与责任明确,详细讲解安装配置、基础语法及SQLit... 目录0、惨痛教训1、SQLite3环境配置(1)、下载安装SQLite库(2)、解压下载的文件(3)、

使用Python绘制3D堆叠条形图全解析

《使用Python绘制3D堆叠条形图全解析》在数据可视化的工具箱里,3D图表总能带来眼前一亮的效果,本文就来和大家聊聊如何使用Python实现绘制3D堆叠条形图,感兴趣的小伙伴可以了解下... 目录为什么选择 3D 堆叠条形图代码实现:从数据到 3D 世界的搭建核心代码逐行解析细节优化应用场景:3D 堆叠图

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方