zoukankan      html  css  js  c++  java
  • CopyOnWrite容器之一:CopyOnWriteArrayList

    一、CopyOnWriteArrayList简介

      为了维护对象的一致性快照,要依靠不可变性(immutability)来消除在协调读取不同的但是相关的属性时需要的同步。对于集合,这意味着如果有大量的读(即get() ) 和迭代,不必同步操作以照顾偶尔的写(即 add() )调用。对于新的 CopyOnWriteArrayListCopyOnWriteArraySet 类,所有可变的(mutable)操作都首先取得后台数组的副本,对副本进行更改,然后替换副本。这种做法保证了在遍历自身更改的集合时,永远不会抛出ConcurrentModificationException。遍历集合会用原来的集合完成,而在以后的操作中使用更新后的集合。

      Copy-On-Write简称COW,是一种用于程序设计中的优化策略。其基本思路是,从一开始大家都在共享同一个内容,当某个人想要修改这个内容的时候,才会真正把内容Copy出去形成一个新的内容然后再改,这是一种延时懒惰策略。从JDK1.5开始Java并发包里提供了两个使用CopyOnWrite机制实现的并发容器,它们是CopyOnWriteArrayList和CopyOnWriteArraySet。CopyOnWrite容器非常有用,可以在非常多的并发场景中使用到。
    什么是CopyOnWrite容器
    CopyOnWrite容器即写时复制的容器。通俗的理解是当我们往一个容器添加元素的时候,不直接往当前容器添加,而是先将当前容器进行Copy,复制出一个新的容器,然后新的容器里添加元素,添加完元素之后,再将原容器的引用指向新的容器。这样做的好处是我们可以对CopyOnWrite容器进行并发的读,而不需要加锁,因为当前容器不会添加任何元素。所以CopyOnWrite容器也是一种读写分离的思想,读和写不同的容器。
    CopyOnWrite容器有很多优点,但是同时也存在两个问题,即内存占用问题和数据一致性问题。所以在开发的时候需要注意一下。
    内存占用问题。因为CopyOnWrite的写时复制机制,所以在进行写操作的时候,内存里会同时驻扎两个对象的内存,旧的对象和新写入的对象(注意:在复制的时候只是复制容器里的引用,只是在写的时候会创建新对象添加到新容器里,而旧容器的对象还在使用,所以有两份对象内存)。如果这些对象占用的内存比较大,比如说200M左右,那么再写入100M数据进去,内存就会占用300M,那么这个时候很有可能造成频繁的Yong GC和Full GC。之前我们系统中使用了一个服务由于每晚使用CopyOnWrite机制更新大对象,造成了每晚15秒的Full GC,应用响应时间也随之变长。
    针对内存占用问题,可以通过压缩容器中的元素的方法来减少大对象的内存消耗,比如,如果元素全是10进制的数字,可以考虑把它压缩成36进制或64进制。或者不使用CopyOnWrite容器,而使用其他的并发容器,如ConcurrentHashMap。
    数据一致性问题。CopyOnWrite容器只能保证数据的最终一致性,不能保证数据的实时一致性。所以如果你希望写入的的数据,马上能读到,请不要使用CopyOnWrite容器。

    二、CopyOnWriteArrayList源码分析

    2.1、类图结构

    2.2、数据结构

    数组存储元素。

    2.3、CopyOnWriteArrayList中的lock

    final transient ReentrantLock lock = new ReentrantLock();

    2.4、成员变量

    private transient volatile Object[] array;
    private static final sun.misc.Unsafe UNSAFE;
    private static final long lockOffset;

    2.5、构造函数

        public CopyOnWriteArrayList() {
            setArray(new Object[0]);
        }
        final void setArray(Object[] a) {
            array = a;
        }

    2.6、增加元素

        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();
            }
        }

    2.7、remove

        public E remove(int index) {
            final ReentrantLock lock = this.lock;
            lock.lock();
            try {
                Object[] elements = getArray();
                int len = elements.length;
                E oldValue = get(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 oldValue;
            } finally {
                lock.unlock();
            }
        }

    2.8、查询元素


    读的时候不需要加锁,如果读的时候有多个线程正在向CopyOnWriteArrayList添加数据,读还是会读到旧的数据,因为写的时候不会锁住旧的CopyOnWriteArrayList。

        public E get(int index) {
            return get(getArray(), index);
        }


    2.9、size

    没有加锁,可能会是旧的数据。

        public int size() {
            return getArray().length;
        }

    2.10、clear

        public void clear() {
            final ReentrantLock lock = this.lock;
            lock.lock();
            try {
                setArray(new Object[0]);
            } finally {
                lock.unlock();
            }
        }


    三、JDK或开源框架中使用

    如上面的分析CopyOnWriteArrayList表达的一些思想:
    1、读写分离,读和写分开
    2、最终一致性
    3、使用另外开辟空间的思路,来解决并发冲突
    通过上面的分析,CopyOnWriteArrayList 有几个缺点:
    1、由于写操作的时候,需要拷贝数组,会消耗内存,如果原数组的内容比较多的情况下,可能导致young gc或者full gc
    2、不能用于实时读的场景,像拷贝数组、新增元素都需要时间,所以调用一个set操作后,读取到数据可能还是旧的,虽然CopyOnWriteArrayList 能做到最终一致性,但是还是没法满足实时性要求;
    网上的文章都说到,CopyOnWriteArrayList 合适读多写少的场景,比如说缓存。
    但是我个人认为CopyOnWriteArrayList 无用武之地,那怕读远远大于写也不能使用CopyOnWriteArrayList,因为谁也没法保证CopyOnWriteArrayList 到底要放置多少数据,万一数据稍微有点多,每次add/set都要重新复制数组,这个代价实在太高昂了。在高性能的互联网应用中,这种操作分分钟引起故障。

    四、示例

    package com.dxz.concurrent.cow;
    
    import java.util.List;
    
    public class ReadThread implements Runnable {
        private List<Integer> list;
    
        public ReadThread(List<Integer> list) {
            this.list = list;
        }
    
        @Override
        public void run() {
            System.out.print("size:="+list.size()+",::");
            for (Integer ele : list) {
                System.out.print(ele + ",");
            }
            System.out.println();
        }
    }
    
    package com.dxz.concurrent.cow;
    
    import java.util.List;
    
    public class WriteThread implements Runnable {
        private List<Integer> list;
    
        public WriteThread(List<Integer> list) {
            this.list = list;
        }
    
        @Override
        public void run() {
            this.list.add(9);
        }
    }
    
    package com.dxz.concurrent.cow;
    import java.util.Arrays;
    import java.util.List;
    import java.util.concurrent.CopyOnWriteArrayList;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    import java.util.concurrent.TimeUnit;
    
    public class TestCopyOnWriteArrayList {
    
        private void test() {
            //1、初始化CopyOnWriteArrayList
            List<Integer> tempList = Arrays.asList(new Integer [] {1,2});
            CopyOnWriteArrayList<Integer> copyList = new CopyOnWriteArrayList<>(tempList);
    
    
            //2、模拟多线程对list进行读和写
            ExecutorService executorService = Executors.newFixedThreadPool(10);
            executorService.execute(new ReadThread(copyList));
            executorService.execute(new WriteThread(copyList));
            executorService.execute(new WriteThread(copyList));
            executorService.execute(new WriteThread(copyList));
            executorService.execute(new ReadThread(copyList));
            executorService.execute(new WriteThread(copyList));
            executorService.execute(new ReadThread(copyList));
            executorService.execute(new WriteThread(copyList));
            try {
                TimeUnit.SECONDS.sleep(5);
            } catch (InterruptedException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
            System.out.println("copyList size:"+copyList.size());
        }
    
    
        public static void main(String[] args) {
            new TestCopyOnWriteArrayList().test();
        }
    }

    结果:

    size:=2,::1,2,9,9,
    size:=5,::1,2,9,9,9,
    size:=5,::1,2,9,9,9,9,
    copyList size:7
  • 相关阅读:
    Android学习(二)
    密码与安全新技术专题之AI与密码
    Android学习(一)
    冲刺周六The Sixth Day(6.1)
    冲刺周日The Seventh Day(6.2)
    冲刺周五The Fifth Day(5.31)
    冲刺周四The Fourth Day(5.30)
    冲刺周三The Third Day(5.29)
    冲刺周二The Second Day(5.28)
    冲刺周一The First Day(5.27)
  • 原文地址:https://www.cnblogs.com/duanxz/p/2718032.html
Copyright © 2011-2022 走看看