直接干:
> 喜欢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
基本这么多吧。拜
网友评论