`
BrokenDreams
  • 浏览: 260029 次
  • 性别: Icon_minigender_1
  • 来自: 北京
博客专栏
68ec41aa-0ce6-3f83-961b-5aa541d59e48
Java并发包源码解析
浏览量:103857
社区版块
存档分类
最新评论

Jdk1.6 JUC源码解析(23)-CopyOnWriteArrayList、CopyOnWriteArraySet

阅读更多

Jdk1.6 JUC源码解析(23)-CopyOnWriteArrayList、CopyOnWriteArraySet

作者:大飞

 

功能简介:
  • CopyOnWriteArrayList是一种线程安全的ArrayList。顾名思义,有写操作时,就会copy一个新的内部数组出来替换掉旧的数组。这样做的好处是,遍历操作不用加锁了,但是遍历的数组不会感知即时变更,只是一个快照。在某些场景下,这要比不用CopyOnWrite,读写都加锁的实现方式要高效一些。CopyOnWriteArrayList一般使用在读多写少的场景。
  • CopyOnWriteArraySet由内部的一个CopyOnWriteArrayList来代理实现。
 
源码分析:
  • 先看下CopyOnWriteArrayList的内部结构:
public class CopyOnWriteArrayList<E>
    implements List<E>, RandomAccess, Cloneable, java.io.Serializable {
    private static final long serialVersionUID = 8673264195747942595L;
    /** The lock protecting all mutators */
    transient final ReentrantLock lock = new ReentrantLock();
    /** The array, accessed only via getArray/setArray. */
    private volatile transient 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;
    }
       内部结构一目了然,一个锁,一个数组。注意数组由volatile修饰。内部数组只能通过getArray和setArray来操作。
 
  • 继续看下CopyOnWriteArrayList的"写"系列方法:
    public E set(int index, E element) {
	    final ReentrantLock lock = this.lock;
	    lock.lock();
	    try {
	        Object[] elements = getArray();
	        Object oldValue = elements[index];
            //新值和旧值做对比。
	        if (oldValue != element) {
                //如果不相等,需要从旧数组拷贝一份新数组,然后设置新值,设置为内部数组。
		        int len = elements.length;
		        Object[] newElements = Arrays.copyOf(elements, len);
		        newElements[index] = element;
		        setArray(newElements);
	        } else {
		        //这里保证volatile写的语义。
		        setArray(elements);
	        }
	        return (E)oldValue;
	    } finally {
	        lock.unlock();
	    }
    }

 

    public boolean add(E e) {
	    final ReentrantLock lock = this.lock;
	    lock.lock();
	    try {
	        Object[] elements = getArray();
	        int len = elements.length;
            //拷贝一份新数组,同时增加一个单位的长度,设置新值,设置为内部数组。
	        Object[] newElements = Arrays.copyOf(elements, len + 1);
	        newElements[len] = e;
	        setArray(newElements);
	        return true;
	    } finally {
	        lock.unlock();
	    }
    }

 

    public void add(int index, E element) {
	    final ReentrantLock lock = this.lock;
	    lock.lock();
	    try {
	        Object[] elements = getArray();
	        int len = elements.length;
	        if (index > len || index < 0)
		        throw new IndexOutOfBoundsException("Index: "+index+
						    ", Size: "+len);
	        Object[] newElements;
	        int numMoved = len - index;
	        if (numMoved == 0)//如果是往数组末尾添加元素,直接拷贝,同时增加一个单位的长度。
		        newElements = Arrays.copyOf(elements, len + 1);
	        else {//如果不是往数组末尾插入一个元素,分两段拷贝。
		        newElements = new Object[len + 1];
		        System.arraycopy(elements, 0, newElements, 0, index);
		        System.arraycopy(elements, index, newElements, index + 1, numMoved);
	        }
            //设置新值,设置为内部数组。
	        newElements[index] = element;
	        setArray(newElements);
	    } finally {
	        lock.unlock();
	    }
    }

 

    public E remove(int index) {
	    final ReentrantLock lock = this.lock;
	    lock.lock();
	    try {
	        Object[] elements = getArray();
	        int len = elements.length;
	        Object oldValue = elements[index];
	        int numMoved = len - index - 1;
	        if (numMoved == 0)//如果要删除的是数组末尾的元素,直接拷贝,同时减少一个单位的长度。然后设置为内部数组。
		        setArray(Arrays.copyOf(elements, len - 1));
	        else {//如果不是删除数组末尾的元素,分两段拷贝,然后设置为内部数组。
		        Object[] newElements = new Object[len - 1];
		        System.arraycopy(elements, 0, newElements, 0, index);
		        System.arraycopy(elements, index + 1, newElements, index, numMoved);
		        setArray(newElements);
	        }
	        return (E)oldValue;
	    } finally {
	        lock.unlock();
	    }
    }

 

    public boolean remove(Object o) {
	    final ReentrantLock lock = this.lock;
	    lock.lock();
	    try {
	        Object[] elements = getArray();
	        int len = elements.length;
	        if (len != 0) {
		        // 假设数组中存在o,那么删除后的长度是len - 1,所以先处理len - 1个元素
		        int newlen = len - 1;
                // 建立一个新数组,长度为len - 1
		        Object[] newElements = new Object[newlen];
		        for (int i = 0; i < newlen; ++i) {
		            if (eq(o, elements[i])) {
			            // 发现了目标元素,跳过这个元素拷贝剩下的元素到新数组。
			            for (int k = i + 1; k < len; ++k)
			                newElements[k-1] = elements[k];
			            setArray(newElements);
			            return true;
		            } else
			            newElements[i] = elements[i];
		        }
		        // 如果到这里的话,说明前len - 1个元素中都没找到o,那么看看最后一个元素是否为o。
		        if (eq(o, elements[newlen])) {
                    //如果是,直接设置新数组就可以了,因为最后一个元素不在新数组里。
		            setArray(newElements);
		            return true;
		        }
	        }
	        return false;
	    } finally {
	        lock.unlock();
	    }
    }
       可见,所有的写操作都会拷贝另外一份内部数组出来,然后替换之前的内部数组,这里也会存在一个内存占用的问题,使用的时候应当注意。其他的写操作都很容易看懂,这里不罗嗦了。
 
  • 读操作很简答:
    public E get(int index) {
        return (E)(getArray()[index]);
    }

 

  • 再看下迭代器相关
    public Iterator<E> iterator() {
        return new COWIterator<E>(getArray(), 0);
    }

    public ListIterator<E> listIterator() {
        return new COWIterator<E>(getArray(), 0);
    }
    private static class COWIterator<E> implements ListIterator<E> {
        /** Snapshot of the array **/
        private final Object[] snapshot;
        /** Index of element to be returned by subsequent call to next.  */
        private int cursor;
        private COWIterator(Object[] elements, int initialCursor) {
            cursor = initialCursor;
            snapshot = elements;
        }
        public boolean hasNext() {
            return cursor < snapshot.length;
        }
        public boolean hasPrevious() {
            return cursor > 0;
        }
        public E next() {
	    if (! hasNext())
                throw new NoSuchElementException();
	    return (E) snapshot[cursor++];
        }
        public E previous() {
	    if (! hasPrevious())
                throw new NoSuchElementException();
	    return (E) snapshot[--cursor];
        }
        public int nextIndex() {
            return cursor;
        }
        public int previousIndex() {
            return cursor-1;
        }

        public void remove() {
            throw new UnsupportedOperationException();
        }

        public void set(E e) {
            throw new UnsupportedOperationException();
        }

        public void add(E e) {
            throw new UnsupportedOperationException();
        }
    }
       注意创建迭代器的时候将当时的array赋予迭代器内部的域,相当于是获取了一个快照,所以遍历的时候不需要加锁,但不支持迭代器上的写操作(写方法都会抛出异常)。
 
  • 没有涉及到的部分,代码都比较容易理解,这里不说了。最后看下CopyOnWriteArraySet,代码很简单:
public class CopyOnWriteArraySet<E> extends AbstractSet<E>
        implements java.io.Serializable {
    private static final long serialVersionUID = 5457747651344034263L;
    private final CopyOnWriteArrayList<E> al;

    public CopyOnWriteArraySet() {
        al = new CopyOnWriteArrayList<E>();
    }

    public CopyOnWriteArraySet(Collection<? extends E> c) {
        al = new CopyOnWriteArrayList<E>();
        al.addAllAbsent(c);
    }

    public int size() {
	    return al.size();
    }

    public boolean isEmpty() {
	    return al.isEmpty();
    }

    public boolean contains(Object o) {
	    return al.contains(o);
    }

    public Object[] toArray() {
	    return al.toArray();
    }

    public <T> T[] toArray(T[] a) {
	    return al.toArray(a);
    }

    public void clear() {
        al.clear();
    }

    public boolean remove(Object o) {
	    return al.remove(o);
    }

    public boolean add(E e) {
	    return al.addIfAbsent(e);
    }
    ...
       内部一个CopyOnWriteArrayList,方法都由内部的CopyOnWriteArrayList来实现。
 
 

       ok,代码解析完毕! 

 

       

       参见:Jdk1.6 JUC源码解析(7)-locks-ReentrantLock

 

 

 

分享到:
评论

相关推荐

    thread_analysis:JDK中JUC学习记录

    本篇将深入解析JDK中的JUC包,探讨其核心组件和机制,帮助开发者更好地理解和利用这些高级并发工具。 一、线程池ExecutorService ExecutorService是JUC包中的核心接口,它扩展了Executor接口,提供了管理和控制线程...

    Java面试宝典2018-最全面试资料

    - **并发集合**:ConcurrentHashMap、CopyOnWriteArrayList和CopyOnWriteArraySet等线程安全的集合。 5. **IO流** - **文件操作**:File类的使用,以及FileInputStream和FileOutputStream进行文件读写。 - **流...

    汪文君高并发编程实战视频资源下载.txt

    │ 高并发编程第二阶段46讲、ClassLoader链接阶段(验证,准备,解析)过程详细介绍.mp4 │ 高并发编程第二阶段47讲、ClassLoader初始化阶段详细介绍clinit.mp4 │ 高并发编程第二阶段48讲、JVM内置三大类加载器...

    汪文君高并发编程实战视频资源全集

    │ 高并发编程第二阶段46讲、ClassLoader链接阶段(验证,准备,解析)过程详细介绍.mp4 │ 高并发编程第二阶段47讲、ClassLoader初始化阶段详细介绍clinit.mp4 │ 高并发编程第二阶段48讲、JVM内置三大类加载器...

    物联网_Android蓝牙双模开发_基于BLE40和BLE50的双通道蓝牙通信框架_支持多设备并发管理与自定义数据包解析的智能硬件控制SDK_适用于智能家居和工业物联网场景的蓝.zip

    物联网_Android蓝牙双模开发_基于BLE40和BLE50的双通道蓝牙通信框架_支持多设备并发管理与自定义数据包解析的智能硬件控制SDK_适用于智能家居和工业物联网场景的蓝

    thymeleaf-2.0.1.jar中文-英文对照文档.zip

    # 压缩文件中包含: 中文-英文对照文档 jar包下载地址 Maven依赖 Gradle依赖 源代码下载地址 # 本文件关键字: jar中文-英文对照文档.zip,java,jar包,Maven,第三方jar包,组件,开源组件,第三方组件,Gradle,中文API文档,手册,开发手册,使用手册,参考手册 # 使用方法: 解压最外层zip,再解压其中的zip包,双击 【index.html】 文件,即可用浏览器打开、进行查看。 # 特殊说明: ·本文档为人性化翻译,精心制作,请放心使用。 ·只翻译了该翻译的内容,如:注释、说明、描述、用法讲解 等; ·不该翻译的内容保持原样,如:类名、方法名、包名、类型、关键字、代码 等。 # 温馨提示: (1)为了防止解压后路径太长导致浏览器无法打开,推荐在解压时选择“解压到当前文件夹”(放心,自带文件夹,文件不会散落一地); (2)有时,一套Java组件会有多个jar,所以在下载前,请仔细阅读本篇描述,以确保这就是你需要的文件;

    在线教育_微信小程序云开发_课程预约管理系统_基于腾讯云开发的智能在线课程预约平台_包含课程时间管理_学员自助预约_多维度签到核销_数据导出分析_适用于教育培训机构_健身房_企业内.zip

    在线教育_微信小程序云开发_课程预约管理系统_基于腾讯云开发的智能在线课程预约平台_包含课程时间管理_学员自助预约_多维度签到核销_数据导出分析_适用于教育培训机构_健身房_企业内

    基于改进二进制粒子群算法的含需求响应机组组合问题研究——MATLAB代码实现及复现国际学术研究A Modified Binary PSO在微电网中的创新应用

    内容概要:本文详细介绍了如何使用MATLAB实现改进二进制粒子群算法来解决含需求响应的机组组合问题。文章首先构建了机组组合问题的基本模型,定义了机组的数量及其最小和最大发电功率等参数。接着,在此基础上加入了需求响应机制,通过设置需求响应的最大可削减负荷量和补偿系数,实现了对负荷侧的管理。为了提高求解效率,采用了一种改进的二进制粒子群算法,该算法通过独特的更新策略使粒子更快更准地找到最优解。此外,还讨论了微电网调度中如何降低发电成本并确保供电稳定性,强调了需求响应在优化调度中的重要性。 适合人群:对电力系统优化、机组组合问题、需求响应机制以及改进粒子群算法感兴趣的科研人员和技术开发者。 使用场景及目标:适用于需要优化电力系统调度的研究项目或实际工程应用,旨在通过引入需求响应机制和改进的粒子群算法,降低发电成本,提高系统的灵活性和经济效益。 其他说明:文中提供的MATLAB代码示例有助于理解和复现算法的具体实现,同时提醒了一些常见的陷阱和注意事项,如负荷预测误差的影响和合理的参数选择。

    chromedriver-linux64-138.0.7153.0.zip

    chromedriver-linux64-138.0.7153.0.zip

    基干领航跟随者与滑膜控制的多智能体编队仿真控制:700行Matlab代码实现变换队形避障

    内容概要:本文详细介绍了基于领航跟随者和滑膜控制的多智能体编队仿真控制系统。首先阐述了领航跟随者策略的工作原理,即由一个领航者决定编队的整体运动方向和其他跟随者根据相对位置进行调整。接着解释了滑膜控制的作用及其在编队控制中的优势,特别是在应对环境干扰和实现精准轨迹跟踪方面的能力。文中提供了具体的Matlab代码示例,涵盖智能体初始化、障碍物检测与避障机制、领航者轨迹生成、滑膜控制器的设计以及队形变换等多个重要环节。此外,作者还分享了一些实用的经验和技术细节,如数值积分步长选择、避障算法优化等。 适合人群:对多智能体系统、编队控制、滑膜控制感兴趣的科研人员、研究生及工程师。 使用场景及目标:适用于需要理解和实现多智能体编队控制系统的场合,特别是涉及无人机编队飞行、自动驾驶车辆协同作业等领域。目标是帮助读者掌握领航跟随者策略和滑膜控制的具体实现方法,提高编队系统的稳定性和灵活性。 其他说明:文中提到的代码可以在Matlab环境下运行,提供了完整的编队控制流程演示,有助于加深对相关概念的理解。同时,文中讨论的一些优化措施也为进一步的研究提供了有价值的参考。

    常见分布函数的拟合与KS检验:Matlab边缘分布拟合与检验方法

    内容概要:本文详细介绍了如何在Matlab中进行常见分布函数的拟合及其KS检验。首先概述了几种常见的概率分布,如正态分布、对数正态分布、伽马分布、威布尔分布、指数分布、瑞利分布和极值分布等。然后通过具体的Matlab代码展示了如何生成示例数据,并利用fitdist函数对这些分布进行拟合。接下来,使用kstest函数进行了KS检验,以评估数据是否符合所拟合的分布。文中还强调了KS检验的局限性和注意事项,如参数估计的影响和p值的解释。最后,通过可视化手段进一步验证了拟合的效果。 适合人群:具备一定数学和编程基础的数据分析师、科研工作者及学生。 使用场景及目标:适用于需要对数据进行概率分布拟合和检验的实际应用场景,如金融风险评估、可靠性分析、信号处理等。目标是帮助读者掌握Matlab中分布拟合和KS检验的具体实现方法,提高数据分析能力。 其他说明:文章不仅提供了详细的代码示例,还讨论了实际应用中的注意事项,如参数估计质量、p值解释和物理意义的选择。同时,强调了KS检验作为参考工具的作用,不应过分依赖统计检验结果,还需结合业务背景做出最终判断。

    MAXHUB-V1.5.3-PRO-setup

    maxhub传屏软件

    基于 ESP32 和 GC9D01 0.71''TFT 的逼真眼睛与写轮眼绘制

    GC9D01_Rotation.h

    物联网_蓝牙50_BLE低功耗通信_基于ESP32芯片的智能家居设备无线控制系统_实现多设备互联与远程监控的跨平台解决方案_包含Android和iOS移动端应用开发_支持MQTT.zip

    物联网_蓝牙50_BLE低功耗通信_基于ESP32芯片的智能家居设备无线控制系统_实现多设备互联与远程监控的跨平台解决方案_包含Android和iOS移动端应用开发_支持MQTT

    iOS开发_AVFoundation框架_Swift编程语言_本地音频播放_网络音频流媒体_后台播放功能_耳机线控支持_UI界面设计_音乐播放器Demo_项目结构优化_错误处理机制.zip

    iOS开发_AVFoundation框架_Swift编程语言_本地音频播放_网络音频流媒体_后台播放功能_耳机线控支持_UI界面设计_音乐播放器Demo_项目结构优化_错误处理机制

    DT决策树回归预测MATLAB代码详解:清晰注释,轻松读取EXCEL数据,适合初学者上手使用

    内容概要:本文详细介绍了如何使用MATLAB实现决策树回归模型,具体步骤包括数据读取、数据预处理、模型训练、预测与评估以及结果可视化。文中提供了完整的代码示例,并针对每个步骤给出了详细的注释和解释,确保读者能够轻松理解和上手。此外,还讨论了决策树回归模型的优点和局限性,如解释性强但易过拟合的问题,并给出了一些优化建议。 适合人群:具备基本编程技能并对机器学习感兴趣的初学者,尤其是希望通过MATLAB快速入门决策树回归模型的人。 使用场景及目标:适用于需要进行回归预测的实际项目,如房价预测、温度预测等。通过学习本文,读者可以掌握如何利用MATLAB实现决策树回归模型,理解模型的工作原理及其应用场景。 其他说明:文中强调了代码实现过程中的一些注意事项,如数据格式、路径设置等,并提供了一些常见的解决方案和优化技巧,帮助读者更好地应对实际问题。

    chromedriver-win64-138.0.7158.0.zip

    chromedriver-win64-138.0.7158.0.zip

    基于储能电站服务的冷热电多微网系统双层优化配置的MATLAB实现与仿真分析

    内容概要:本文详细介绍了基于储能电站服务的冷热电多微网系统的双层优化配置方法及其MATLAB代码实现。文章首先阐述了冷热电多微网系统的重要性和储能电站在其中的关键作用,随后讲解了双层优化配置的概念,即上层优化关注系统宏观目标(如成本最小化、能源利用率最大化),下层优化侧重于各微网系统的具体运行情况。文中提供了具体的MATLAB代码示例,包括上层和下层优化的实现方式,并通过实例展示了如何使用CPLEX求解器进行优化。此外,文章还探讨了模型的实际应用效果,指出相比于传统单层优化,双层优化能够使系统运行成本降低15%-20%,特别是在储能的削峰填谷方面表现出色。最后,作者分享了一些调试经验和实用技巧,如延迟约束生成、结构化数据管理和迭代过程中的收敛判断等。 适用人群:从事能源系统优化的研究人员和技术人员,尤其是对MATLAB和CPLEX有一定了解的读者。 使用场景及目标:适用于需要对冷热电多微网系统进行优化配置的工程项目,旨在通过合理的储能电站配置和调度策略,实现能源利用效率的最大化和系统运行成本的最小化。 其他说明:文章不仅提供了理论指导,还附带了详细的代码实现,便于读者理解和实践。对于初学者而言,建议逐步学习并调试代码,掌握双层优化配置的核心思想和关键技术。

    三菱FX5U七轴标准程序详解:轴点动、回零、相对与绝对定位全面解析,手把手教你编写控制程序,程序流程清晰明了,附带触摸屏操作指南

    内容概要:本文深入解析了三菱FX5U PLC七轴控制系统的标准程序,涵盖主控程序、点动/回零/定位控制、手动模块、复位程序以及生产计数模块。主控程序利用状态寄存器进行模式切换,确保各模块独立运作;点动模式通过PLSY指令控制电机正反转,绝对定位采用DRVA指令,回零则使用ZRN指令;手动模块加入双线圈互锁设计防止误操作;复位程序包含状态检测,确保安全复位;生产计数模块通过上升沿触发避免计数异常。此外,文章还介绍了触摸屏程序的设计要点和调试过程中的一些常见问题及其解决方案。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是对三菱PLC编程有一定基础的人群。 使用场景及目标:适用于需要理解和掌握三菱FX5U PLC七轴控制系统的设计和调试的技术人员。主要目标是帮助读者理解各个功能模块的工作原理,提高编程效率和系统稳定性。 其他说明:文中提供了大量实际应用中的经验和技巧,有助于读者在实际工作中避免常见的错误,提升工作效率。

    【图像分割】基于matlab双目标遗传算法NSGAII优化PCNN图像分割【含Matlab源码 13257期】.zip

    985研究生,Matlab领域优质创作者 (1)如需代码 加腾讯企鹅号,见评论区或私信; (2)代码运行版本 Matlab 2019b (3)其他仿真咨询 1 完整代码包运行+运行有问题可咨询 2 期刊或论文复现; 3 程序定制; 4 期刊写作或指导; 5 科研合作;

Global site tag (gtag.js) - Google Analytics