走进源码——CopyOnWriteArrayList阅读笔记

CopyOnWriteArrayList

无继承,实现了List,RandomAccess,Cloneable,和Serializable接口,具有List的特性,提供可随机访问,提供自身克隆以及序列化的一个容器类。
特点:线程安全;读写分离;数组实现

CopyOnWriteArrayList这个容器类的名字也很好理解,写时拷贝列表,网上对COW(CopyOnWrite)有一个高大上的名字,叫做读写分离,所以CopyOnWriteArrayList是读写分离列表类,下面描述下它的实现

成员变量

相比于ArrayList,CopyOnWriteArrayList的成员变量发生了很多改变,就只有下面几个

// 序列化ID
private static final long serialVersionUID = 8673264195747942595L;
// 锁变量
final transient ReentrantLock lock = new ReentrantLock();
// 真正存储数据的数组
private transient volatile Object[] array;

甚至是size都没了,而size()函数的实现则变成了下面这样

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

首先是对array操作的限制,CopyOnWriteArrayList提供了array的setter和getter方法,在下文会看到,涉及到array的操作,都是通过setter和getter来完成的

final Object[] getArray() {
    return array;
}
final void setArray(Object[] a) {
    array = a;
}

其次是构造函数,CopyOnWriteArrayList提供了缺省,带集合类和带数组的三个构造函数,在这三个函数里面都能看到setArray()的影子。而其中,缺省的构造函数初始化容器的容量为0

public CopyOnWriteArrayList() {
    setArray(new Object[0]);
}
public CopyOnWriteArrayList(Collection<? extends E> c) {
    Object[] elements;
    if (c.getClass() == CopyOnWriteArrayList.class)
        elements = ((CopyOnWriteArrayList<?>)c).getArray();
    else {
        elements = c.toArray();
        // c.toArray might (incorrectly) not return Object[] (see 6260652)
        if (elements.getClass() != Object[].class)
            elements = Arrays.copyOf(elements, elements.length, Object[].class);
    }
    setArray(elements);
}
public CopyOnWriteArrayList(E[] toCopyIn) {
    setArray(Arrays.copyOf(toCopyIn, toCopyIn.length, Object[].class));
}

接着看最重头戏的add()方法,

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

我们可以看到,CopyOnWriteArrayList没有像ArrayListVector那样复杂的扩容机制,只通过两句代码即实现了添加数据,又实现了扩容

  Object[] newElements = Arrays.copyOf(elements, len + 1);
  newElements[len] = e;
  setArray(newElements);

接着,在方法的开始和结束分别是上锁lock.lock();和解锁lock.unlock();,给方法上锁的一个原因是避免多线程环境下,多个线程拷贝出多个副本,然后多个副本操作数据后影响数据的正确性。
除了add()方法外,set(),remove()等的操作数据的操作,基本都和add()方法类似,这里就不在赘述了。
再接着看获取的方法get()

private E get(Object[] a, int index) {
    return (E) a[index];
}

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

可以看到,get()方法是直接在array中进行获取。
综合看add()get()方法,add()方法在进行的时候会先上锁,然后拷贝一份数据的副本进行操作,然后将操作后的副本赋值到array中去,至于get()方法是直接获取array中的值,也正是这样,实现了读写分离。但是这也会有一个问题,就是数据的实时不一致性,有可能副本还没赋值到array,而用户通过get()方法获取了旧的array的值。

来看下CopyOnWriteArrayList的迭代器,因为它没有继承AbstractList因此也没有modCount这个变量,先看下怎么得到它的迭代器

public ListIterator<E> listIterator(int index) {
    Object[] elements = getArray();
    int len = elements.length;
    if (index < 0 || index > len)
        throw new IndexOutOfBoundsException("Index: "+index);

    return new COWIterator<E>(elements, index);
}

private COWIterator(Object[] elements, int initialCursor) {
    cursor = initialCursor;
    snapshot = elements;
}

可以看到是直接通过array和初始化的下标来获取到迭代器。至于迭代数据则是直接获取

@SuppressWarnings("unchecked")
public E next() {
   if (! hasNext())
       throw new NoSuchElementException();
       return (E) snapshot[cursor++];
}

因此,这里没有像ArrayList那样,有一个checkForComodification,检查是否迭代数据被修改,因此我又测试了下

意料之中的毫无错误

ArrayList中,如果获取到迭代器之后对容器内数据进行操作,就会报ConcurrentModificationException,而在CopyOnWriteArrayList则不存在这个问题。
但是也有另外一个问题,如图片所展示那样,获取到迭代器之后修改容器内的数据,拦截器并不会有所更新,要获取到修改后的数据,还得重新获取迭代器。

剩下的一些方法就不赘述啦,大同小异。

总结

CopyOnWriteArrayList的核心是在操作数据的时候,通过拷贝原容器数组的副本,在副本上进行操作后,将副本赋值到容器数组,从而实现读写分离。
CopyOnWriteArrayList适用于读多写少的场景,因为写的时候会拷贝数组,很容易产生大量垃圾而触发GC,不利于系统的性能。

水平有限,难免有错,还请评论区指正。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,015评论 11 349
  • Java SE 基础: 封装、继承、多态 封装: 概念:就是把对象的属性和操作(或服务)结合为一个独立的整体,并尽...
    Jayden_Cao阅读 2,048评论 0 8
  • 旧文一篇 2010年最后的时间留给了白色。纯洁的世界给了很多人想像,也带给了很多人快乐。看那雪地上滞留的孩子,看那...
    子凌阅读 398评论 2 3
  • 陳湘阅读 151评论 0 0
  • 哑巴,曾经是小镇上的一个人,因为不会说话,所以大家都叫他哑巴,至于真名叫什么,没有几个人清楚。 哑巴是怎么哑的也没...
    韩毅阅读 650评论 4 5