转载请注明出处:http://blog.csdn.net/hjf_huangjinfu/article/details/64123255
Vector
Vector 作为 List 接口的实现类,底层也是采用 数组 来存储数据,和 ArrayList 一样(准确说是 ArrayList 与 Vector 一样,毕竟是先有 Vector 后有 ArrayList)。
但是 Vector 和 ArrayList 的不同的是:Vector 是线程安全的,对外提供的读写方法,都是 synchronized 方法,有个别 非 synchronized 方法,其实它内部也是 synchronized 的,比如:
public void add(int index, E element) {
insertElementAt(element, index);
}
public synchronized void insertElementAt(E obj, int index) {
modCount++;
if (index > elementCount) {
throw new ArrayIndexOutOfBoundsException(index
+ " > " + elementCount);
}
ensureCapacityHelper(elementCount + 1);
System.arraycopy(elementData, index, elementData, index + 1, elementCount - index);
elementData[index] = obj;
elementCount++;
}
而 ArrayList 是线程不安全的,当不涉及到多线程时,ArrayList 效率会高于 Vector。
关于 ArrayList 详情,参考:Java 中 ArrayList 的实现解析
Stack
Stack 实现了一个 LIFO 的栈的功能,他是继承于 Vector 来实现的,所以内部也是数组。
示例图:
CopyOnWriteArrayList
CopyOnWriteArrayList 作为 List 接口的实现类,在 jdk 1.5 被引入,其目的是提供一个 线程安全(相对于 ArrayList 等非线程安全的 List) 并且 高效(相对于同样线程安全的 Vector) 的 List。
线程安全
CopyOnWriteArrayList 的内部实现机制正如它的名字一样:
Copy-on-Write,也就是“写时复制”,当有 写类型的操作作用到 CopyOnWriteArrayList 对象的时候,它们都会先获取锁,然后复制一份当前数据作为副本,然后在当前的数据副本上做修改,最后把修改提交,然后释放锁。
Array-List,也就是底层是使用数组(顺序表)来进行数据存储,大体实现机制和 ArrayList 差不多。
代码示例(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 比 Vector 高效,主要有以下 2 个原因:
1、Vector 中,读写操作都被加锁了,而 CopyOnWriteArrayList 中,只有写操作才被加锁,而读操作没有进行加锁。这样,当 读 线程数量远大于 写 线程数量的时候,CopyOnWriteArrayList 尤为高效。
2、Vector 中,使用的是 内置锁(内置锁参考:Java内置锁的简单认识),而 CopyOnWriteArrayList 中,使用的是 jdk 1.5 引入的 ReentrantLock ,相比于 内置锁,ReentrantLock 的性能还是有所提升的。