CopyOnWriteArrayList

1、空参构造函数

public CopyOnWriteArrayList() {
    setArray(new Object[0]);
}

构造一个空数组,调用setArray将其赋值给成员变量array。

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

成员变量array用volatile修饰。

private transient volatile Object[] array;

2、CopyOnWrite的写时复制的机制

1)新增

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

当前数组复制到了新的数组里去,长度加1。

setArray(newElements);

最后再把新的数组设置为CopyOnWriteArrayList对应的一个数组,volatile保证,只要他一写,其他线程可以立马读到
2)更新

Object[] newElements = Arrays.copyOf(elements, len);

然后

setArray(newElements);

3)删除
稍微复杂一点
1)如果你要删除的是最后一个元素,是不需要移动任何元素

setArray(Arrays.copyOf(elements, len - 1));

2)如果是中间的元素,则需要分两次复制过去。

4、总结

1)CopyOnWriteArrayList底层数据结构是Object[]数组。增删改操作的时候,都必须先获取一把ReentrantLock独占锁。CopyOnWriteArrayList的并发写性能不如ConcurrentHashMap。ConcurrentHashMap,并发写CAS非阻塞式的分段加锁,并发读是通过volatile来保证最新。CopyOnWriteArrayList的并发写性能是比较差的,所有线程要写都是串行。
2)ReentrantLock底层基于AQS,AQS底层基于CAS,CAS底层基于硬件级别的锁。
3)CopyOnWriteArrayList不是基于CAS执行读写操作,不需要依赖任何一种加锁的机制来保证数据读写并发的安全性,甚至都不需要依赖于Unsafe.getObjectVolatile()。
4)只有一个线程可以写,但同时可以允许大量的线程来并发读。
5)CopyOnWriteArrayList增删改:独占锁 + 写时复制。
6)弱一致性:写线程和读线程看到的不一致。
7)hdfs的源码就用到了CopyOnWriteArrayList,适用于增删改不多,大量的读操作比如遍历等场景下。

3、缺点

1)空间换时间,内存占用翻倍。
2)弱一致性

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值