java并发copyonwritelist解惑

直接干:

> 喜欢java并发编程的请加群:736156823

1.jdk版本jdk-8u121-linux-x64;

2.根据名字来,写的时候复制,其实是写的时候先加锁防止写的时候出现并发问题,然后加锁成功后在进行复制;

3.读的时候是不需要加锁的,并发的读取数据;

4.为什么写操作的时候加锁?首先上代码,那add说事,其他的写操作类似,源代码433行开始: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();}}以上代码可以看到,不加锁时候,很容易丢失插入的数据,因为可能大家获取的是最初始化的版本数组,所以最后就可能的情况是只插入了一个数据,数组中就一个数据(第一次使用ubuntu写简书,也是第一次写,感觉写的好渣……哈哈)。所以必须枷锁。不多解释了,哼

5.读的时候不用加锁,因为没有改变数据,没有破坏,所以不用枷锁,这个不多说;

6.如果add后,立即get,并发的那种,能马上获得数据吗?可能不会的,原因上代码,源代码441行:setArray(newElements);这一句执行后才真正的存进去了;

7.setArray做了什么?源代码112行开始:final void setArray(Object[] a) {array = a;}这里的等于号,引用的赋值操作的同步由jvm内部解决,而且private transient volatile Object[] array;由volatile修饰,所以在复制完成时直接重定向引用就可以了;

8.这个东西不支持实时性但是会确保最终一致性,请参考上面几点;

9.内部的复制操作使用的是Object[] newElements = Arrays.copyOf(elements, len + 1);和System.arraycopy(elements, 0, newElements, 0, index);其实都是system的copy;

10.然后就是迭代器了,public void remove() {throw new UnsupportedOperationException();}public void set(E e) {throw new UnsupportedOperationException();}public void add(E e) {throw new UnsupportedOperationException();}这三个方法是不支持的;

11.使用场景自然就是读远多于写的情况啦,哈哈,但是有一个问题,那就是初始化问题,我不能上来执行10000次的add操作后然后在读取吧?哈哈,所以,可以先用collection对他进行初始化,之后就可以并发读啦,然后后面有几次或者多次的写也是安全的,然后就用就行了。

> 喜欢java并发编程的请加群:736156823

基本这么多吧。拜

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • ``` /* * ORACLE PROPRIETARY/CONFIDENTIAL. Use is subject ...
    非专业码农阅读 2,701评论 0 0
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,929评论 18 399
  • 对象的创建与销毁 Item 1: 使用static工厂方法,而不是构造函数创建对象:仅仅是创建对象的方法,并非Fa...
    孙小磊阅读 6,210评论 0 3
  • 1.解决信号量丢失和假唤醒 public class MyWaitNotify3{ MonitorObject m...
    Q罗阅读 4,425评论 0 1
  • 缓存一致性问题 计算机在执行程序时,每条指令都是在CPU中执行的,而执行指令过程中,势必涉及到数据的读取和写入。由...
    谜碌小孩阅读 4,268评论 8 3