• CopyOnWriteArrayList源码分析


    CopyOnWriteArrayList源码分析

    作用:CopyOnWriteArrayList(Copy-On-Write: 写入时复制)是一个线程安全的ArrayList,对其进行的修改操作都是先加锁然后在底层的一个复制数组上进行。

    优点:经常被用于"读多写少"的并发场景,因为读取的时候不需要加锁,性能较好。读写分离,在使用迭代器迭代的时候不会抛出异常。

    缺点:需要拷贝原数据,数据较大的时候容易引起频繁Full GC;写和读在不同的数组上,读取的是旧数组的数据。

    1. 成员变量
    2. // 独占锁
    3. final transient ReentrantLock lock = new ReetrantLock(); 
    4. // 只能通过getArray/setArray方法来获取/修改array
    5. private transient volatile Object[] array;

    初始化

    1. 创建空链表

    1. public CopyOnWriteArrayList() {
    2.     setArrayList(new Object[0]);
    3. }


    2. 根据给定的集合数据创建链表

    1. public CopyOnWriteArrayList(Collections c) {
    2.     Object[] elements;
    3.     //如果是CopyOnWriteArrayList类,直接引用
    4.     if (c.getClass() == CopyOnWriteArrayList.class) {
    5.         elements = ((CopyOnWriteArrayList)c).getArray();
    6.     } else {
    7.         elements = c.toArray();
    8.         //c.toArray可能不会返回Object[]类型
    9.         if (elements.getClass() != Object[].class) {
    10.             //通过Arrays.copyOf复制数组
    11.             elements = Arrays.copyOf(elements, elements.length, Object[].class);
    12.         }
    13.     }
    14.     //设置array变量为elements
    15.     setArray(elements);
    16. }

    部分方法源码

    set(int index, E element) 修改array[index]的值

    1. public E set(int index, E element) {
    2.     final ReentrantLock lock = this.lock;
    3.     //加锁
    4.     lock.lock();
    5.     try {
    6.         //拿到成员变量array
    7.         Object[] elements = getArray();
    8.         // 获取elements[index]的值
    9.         E oldValue = get(elements, index);
    10.         //如果值发生了变化
    11.         if (oldValue != element) {
    12.             int len = elements.length;
    13.             //先对elements复制一个拷贝
    14.             Object[] newElements = Arrays.copyOf(elements, len);
    15.             //更新值
    16.             newElements[index] = element;
    17.             //更新this.array = newElements
    18.             setArray(newElements);
    19.         } else {
    20.             //为了保证volatile语义 
    21.             setArray(elements);
    22.         }
    23.         //返回原始值
    24.         return oldValue;
    25.     } finally {
    26.         //释放锁
    27.         lock.unlock();
    28.     }
    29. }


    add(E e) 添加元素到末尾

    1. //和上面的set差不多 流程: 先获取锁->拷贝数组->修改拷贝数组->修改原始数组引用
    2. public boolean add(E e) {
    3.     final ReentrantLock lock = this.lock;
    4.     //加锁
    5.     lock.lock();
    6.     try {
    7.         Object[] elements = getArray();
    8.         int len = elements.length;
    9.         //拷贝数组
    10.         Object[] newElements = Arrays.copyOf(elements, len + 1);
    11.         //添加值
    12.         newElements[len] = e;
    13.         //修改array引用
    14.         setArray(newElements);
    15.         return true;
    16.     } finally {
    17.         //释放锁
    18.         lock.unlock();
    19.     }
    20. }

    add(int index, E element) 按照下标位置添加 

    1. public void add(int index, E element) {
    2. //加锁
    3. final ReentrantLock lock = this.lock;
    4. lock.lock();
    5. try {
    6. //获取原数组并计算长度
    7. Object[] elements = getArray();
    8. int len = elements.length;
    9. //如果下标超出数组长度并小于0抛出下标越界异常
    10. if (index > len || index < 0)
    11. throw new IndexOutOfBoundsException("Index: "+index+
    12. ", Size: "+len);
    13. //新建空数组
    14. Object[] newElements;
    15. int numMoved = len - index;//计算添加位置下标
    16. if (numMoved == 0)
    17. newElements = Arrays.copyOf(elements, len + 1);//末尾直接复制一个长度+1的新数组
    18. else {
    19. newElements = new Object[len + 1];//复制所有除了要添加的元素
    20. System.arraycopy(elements, 0, newElements, 0, index);
    21. System.arraycopy(elements, index, newElements, index + 1,
    22. numMoved);
    23. }
    24. newElements[index] = element;//根据下标添加新元素到新数组
    25. setArray(newElements);
    26. } finally {
    27. lock.unlock();
    28. }
    29. }

    remove(int index) 删除index位置的值

    1. public E remove(int index) {
    2.     final ReentrantLock lock = this.lock;
    3.     //加锁
    4.     lock.lock();
    5.     try {
    6.         Object[] elements = getArray();
    7.         int len = elements.length;
    8.         //index位置的值
    9.         E oldValue = get(elements, index);
    10.         //index之后需要移动的元素个数
    11.         int numMoved = len - index - 1;
    12.         //不需要移动
    13.         if (numMoved == 0)
    14.             setArray(Arrays.copyOf(elements, len - 1));
    15.         else {
    16.             //创建新数组
    17.             Object[] newElements = new Object[len - 1];
    18.             //拷贝删除位置前面的元素
    19.             System.arraycopy(elements, 0, newElements, 0, index);
    20.             //拷贝删除元素后面的元素
    21.             System.arraycopy(elements, index + 1, newElements, index,
    22.                                 numMoved);
    23.             //修改array引用
    24.             setArray(newElements);
    25.         }
    26.         return oldValue;
    27.     } finally {
    28.         lock.unlock();
    29.     }
    30. }

    emoveRange(int fromIndex, int toIndex) 根据范围删除元素 

    1. public void removeRange(int fromIndex, int toIndex) {
    2. final ReentrantLock lock = this.lock;
    3. lock.lock();
    4. try {
    5. Object[] elements = getArray();
    6. int len = elements.length;
    7. // 如果开始下标<0 或者结束下标大于数组长度或者结束下标小于开始下标
    8. // 抛下标越界异常
    9. if (fromIndex < 0 || toIndex > len || toIndex < fromIndex)
    10. throw new IndexOutOfBoundsException();
    11. // 计算新数组长度
    12. int newlen = len - (toIndex - fromIndex);
    13. // 判断结束下标是否是最后一个下标
    14. int numMoved = len - toIndex;
    15. if (numMoved == 0)
    16. //复制原数组
    17. setArray(Arrays.copyOf(elements, newlen));
    18. else {
    19. //新建一个长度为newlen的数组
    20. Object[] newElements = new Object[newlen];
    21. //将除了需要删除的下标元素复制到新数组
    22. System.arraycopy(elements, 0, newElements, 0, fromIndex);
    23. System.arraycopy(elements, toIndex, newElements,
    24. fromIndex, numMoved);
    25. setArray(newElements);
    26. }
    27. } finally {
    28. lock.unlock();
    29. }
    30. }

    retainAll(Collection c) 在当前集合保留传入集合中的元素,当前元素和集合中元素做交集
     

    1. public boolean retainAll(Collection c) {
    2. //集合为null抛出空指针异常
    3. if (c == null) throw new NullPointerException();
    4. final ReentrantLock lock = this.lock;
    5. lock.lock();
    6. try {
    7. //获取当前集合对象中数组
    8. Object[] elements = getArray();
    9. int len = elements.length;
    10. //当前数组不为空
    11. if (len != 0) {
    12. // temp array holds those elements we know we want to keep
    13. int newlen = 0;
    14. //创建中间数组
    15. Object[] temp = new Object[len];
    16. //遍历数组
    17. for (int i = 0; i < len; ++i) {
    18. Object element = elements[i];
    19. //判断集合c中是否包含elements中的元素
    20. if (c.contains(element))
    21. //包含则将元素存入中间数组
    22. temp[newlen++] = element;
    23. }
    24. //中间数组下标+1之后不等于elements数组长度
    25. if (newlen != len) {
    26. //使用newlen作为新数组长度复制
    27. setArray(Arrays.copyOf(temp, newlen));
    28. return true;
    29. }
    30. }
    31. return false;
    32. } finally {
    33. lock.unlock();
    34. }
    35. }

    clear()方法 清空集合

    1. public void clear() {
    2. final ReentrantLock lock = this.lock;
    3. lock.lock();
    4. try {
    5. //指向一个新建长度为0的数组
    6. setArray(new Object[0]);
    7. } finally {
    8. lock.unlock();
    9. }
    10. }

     addAll(int index, Collection c) 按照下标位置批量添加元素

    1. public boolean retainAll(Collection c) {
    2. //集合为null抛出空指针异常
    3. if (c == null) throw new NullPointerException();
    4. final ReentrantLock lock = this.lock;
    5. lock.lock();
    6. try {
    7. //获取当前集合对象中数组
    8. Object[] elements = getArray();
    9. //获取数组的长度
    10. int len = elements.length;
    11. //如果当前数组不为空
    12. if (len != 0) {
    13. // temp array holds those elements we know we want to keep
    14. int newlen = 0;
    15. //创建一个临时数组
    16. Object[] temp = new Object[len];
    17. //遍历数组
    18. for (int i = 0; i < len; ++i) {
    19. Object element = elements[i];
    20. //判断集合c中是否包含elements中的元素
    21. if (c.contains(element))
    22. //包含则将元素存入临时数组
    23. temp[newlen++] = element;
    24. }
    25. //如果临时数组是否等于elements数组长度
    26. if (newlen != len) {
    27. //不等于则使用newlen作为新数组长度复制
    28. setArray(Arrays.copyOf(temp, newlen));
    29. return true;
    30. }
    31. }
    32. return false;
    33. } finally {
    34. lock.unlock();
    35. }
    36. }
  • 相关阅读:
    软著申请注意事项
    【蜂鸟E203内核解析】Chap.3 自定义指令与协处理器设计
    DBCO-NH-(CH2)4COOH, CAS:2375193-74-9
    几何变换 - 图像的缩放、翻转、仿射变换、透视等
    CSDN一站式云服务开放内测,诚邀C站新老用户来抢鲜
    工业智能网关BL110应用之三十二: 如何连接配置阿里云服务器
    痞子衡嵌入式:IAR环境下无法直接下载调试i.MXRT分散链接工程的解决方案(宏文件.mac+双Flashloader)
    电力物联网关智能通讯管理机-安科瑞黄安南
    OpenGL ES EGL eglCreatePbufferSurface
    机器视觉(四):空域图像增强(1)
  • 原文地址:https://blog.csdn.net/qq_51956240/article/details/126923443