首先咱们来看看List接口,由于ArrayList和CopyOnWriteArrayList都是实现了List接口,咱们今天主要是研究增删改查原理,因此只看相应的方法便可。数组
public interface List<E> extends Collection<E> { int size(); boolean isEmpty(); boolean contains(Object o); Iterator<E> iterator(); Object[] toArray(); <T> T[] toArray(T[] a); boolean add(E e); boolean remove(Object o); boolean containsAll(Collection<?> c); boolean addAll(Collection<? extends E> c); boolean addAll(int index, Collection<? extends E> c); boolean removeAll(Collection<?> c); boolean retainAll(Collection<?> c); void clear(); boolean equals(Object o); int hashCode(); E get(int index); E set(int index, E element); void add(int index, E element); E remove(int index); int indexOf(Object o); int lastIndexOf(Object o); ListIterator<E> listIterator(); ListIterator<E> listIterator(int index); List<E> subList(int fromIndex, int toIndex); }
1)增安全
public boolean add(E e) { //进行数组容量判断,不够就扩容 ensureCapacityInternal(size + 1); // Increments modCount!! elementData[size++] = e; return true; } public void add(int index, E element) { //检查是否会越界 rangeCheckForAdd(index); //进行数组容量判断,不够就扩容 ensureCapacityInternal(size + 1); // Increments modCount!! //将index至数据最后一个元素总体日后移动一格,而后插入新的元素 System.arraycopy(elementData, index, elementData, index + 1, size - index); elementData[index] = element; size++; }
2)删并发
public E remove(int index) { //判断是否越界 rangeCheck(index); modCount++; E oldValue = elementData(index); int numMoved = size - index - 1; //若该元素不是最后一个元素的话,将index+1至数组最后一个元素总体向前移动一格 if (numMoved > 0) System.arraycopy(elementData, index+1, elementData, index, numMoved); elementData[--size] = null; // clear to let GC do its work return oldValue; }
3)改性能
public E set(int index, E element) { rangeCheck(index); E oldValue = elementData(index); elementData[index] = element; return oldValue; }
逻辑很简单,将数组对应index的元素进行替换ui
4)查this
public E get(int index) { rangeCheck(index); return elementData(index); } E elementData(int index) {return (E) elementData[index]; }
逻辑很简单,进行数组越界判断,获取数组对应index的元素spa
以上部分就是ArrayList的增删改查原理,以此也能够解答咱们第二个问题了,ArrayList的底层是数组,因此查询的时候直接根据索引能够很快找到对应的元素,改也是如此,找到index对应元素进行替换。而增长和删除就涉及到数组元素的移动,因此会比较慢。code
1)增索引
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(); } } public void add(int index, E element) { final ReentrantLock lock = this.lock; lock.lock(); try { Object[] elements = getArray(); int len = elements.length; if (index > len || index < 0) throw new IndexOutOfBoundsException("Index: "+index+ ", Size: "+len); Object[] newElements; int numMoved = len - index; if (numMoved == 0) newElements = Arrays.copyOf(elements, len + 1); else { newElements = new Object[len + 1]; System.arraycopy(elements, 0, newElements, 0, index); System.arraycopy(elements, index, newElements, index + 1, numMoved); } newElements[index] = element; setArray(newElements); } finally { lock.unlock(); } }
2)删接口
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]; //将index+1至最后一个元素向前移动一格 System.arraycopy(elements, 0, newElements, 0, index); System.arraycopy(elements, index + 1, newElements, index, numMoved); setArray(newElements); } return oldValue; } finally { lock.unlock(); } }
3)改
public E set(int index, E element) { final ReentrantLock lock = this.lock; //得到锁 lock.lock(); try { Object[] elements = getArray(); E oldValue = get(elements, index); if (oldValue != element) { int len = elements.length; //建立新数组 Object[] newElements = Arrays.copyOf(elements, len); //替换元素 newElements[index] = element; //将新数组指向原来的引用 setArray(newElements); } else { // Not quite a no-op; ensures volatile write semantics setArray(elements); } return oldValue; } finally { //释放锁 lock.unlock(); } }
4)查
//直接获取index对应的元素 public E get(int index) {return get(getArray(), index);} private E get(Object[] a, int index) {return (E) a[index];}
从以上的增删改查中咱们能够发现,增删改都须要得到锁,而且锁只有一把,而读操做不须要得到锁,支持并发。为何增删改中都须要建立一个新的数组,操做完成以后再赋给原来的引用?这是为了保证get的时候都能获取到元素,若是在增删改过程直接修改原来的数组,可能会形成执行读操做获取不到数据。
我知道Vector是增删改查方法都加了synchronized,保证同步,可是每一个方法执行的时候都要去得到锁,性能就会大大降低,而CopyOnWriteArrayList 只是在增删改上加锁,可是读不加锁,在读方面的性能就好于Vector,CopyOnWriteArrayList支持读多写少的并发状况。