美文网首页
java并发copyonwritelist解惑

java并发copyonwritelist解惑

作者: 胶布小子 | 来源:发表于2017-03-28 17:17 被阅读0次

    直接干:

    > 喜欢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

    基本这么多吧。拜

    相关文章

      网友评论

          本文标题:java并发copyonwritelist解惑

          本文链接:https://www.haomeiwen.com/subject/ojojottx.html