ArrayList 是咱们经常使用的工具类之一,可是在多线程的状况下,ArrayList 做为共享变量时,并非线程安全的。主要有如下两个缘由:java
若是咱们想在多线程状况下使用 ArrayList 怎么办?有如下几种办法:数组
先来看看 SynchronizedLis,Collections 其实就是对 ArrayList 进行了一个加锁包装,这个从源码中能够看出;安全
...部分源码,完整源码请查看 JDK 源码...
public void add(int index, E element) {
synchronized (mutex) {list.add(index, element);}
}
public E remove(int index) {
synchronized (mutex) {return list.remove(index);}
}
复制代码
对于 Collections.SynchronizedList 比较简单,就是锁包装了一下,就很少说了~多线程
CopyOnWriteArrayList 也是 JUC 下面的一个并发容器类。不知道你发现没有,但凡你经常使用的集合类,在 JUC 下基本上均可以找到一个并发类,好比 hashMap 有对应的 ConcurrentHashMap。架构
CopyOnWriteArrayList 跟 ArrayList 在总体架构上并无什么区别,底层都是基于数组实现的。不一样的地方大概有两点:并发
CopyOnWriteArrayList 的加锁操做跟 Collections.SynchronizedList 简单的加锁还不同,CopyOnWriteArrayList 中的加锁过程仍是很是值得学习的。CopyOnWriteArrayList 的加锁过程,大概能够归纳为如下四步:app
结合源码来深刻了解 CopyOnWriteArrayList 的并发实现,咱们选择 ArrayList 最简单的将元素新增数组尾部的操做来分析实现过程,源码以下:工具
/** * Appends the specified element to the end of this list. * * @param e element to be appended to this list * @return {@code true} (as specified by {@link Collection#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 就是经过加锁来讲实现容器安全的,可能你会有疑问,为何引入一个新数组,数组的拷贝仍是消耗时间的,直接在原数组上操做不就行了吗?。主要缘由有如下两点:学习
ConcurrentModificationException
异常问题。其余的新增方法就本身去查看源码了,相差很少,基本上是同样的。对数组的删除跟新增都是差很少,不一样的地方是在删除了时候,赋值给新数组时会出现不一样的选择策略。我把源码贴上:this
public E remove(int index) {
final ReentrantLock lock = this.lock;
// 加锁
lock.lock();
try {
Object[] elements = getArray();
int len = elements.length;
E oldValue = get(elements, index);
// 先计算出要移动的问题
int numMoved = len - index - 1;
// 根据移动的位置选择策略
if (numMoved == 0)
setArray(Arrays.copyOf(elements, len - 1));
else {
Object[] newElements = new Object[len - 1];
System.arraycopy(elements, 0, newElements, 0, index);
System.arraycopy(elements, index + 1, newElements, index,
numMoved);
setArray(newElements);
}
return oldValue;
} finally {
//解锁
lock.unlock();
}
}
复制代码
CopyOnWriteArrayList 还有其余的方法,在这里我就不过多介绍了。根据大家本身的疑问去扒一扒 CopyOnWriteArrayList 的源码就知道了,整体来讲 CopyOnWriteArrayList 并不难,甚至感受比 ArrayList 要简单。
总结一下:CopyOnWriteArrayList 是安全的并发容器,有如下两个特色:
欢迎关注公众号【互联网平头哥】,一块儿成长,一块儿进步~。