Vector或Collections.synchronizedList为什么get方法要加锁?CopyOnWriteArrayList并没有。
我觉得读的时候不加锁应该没问题吧?读时候加锁有什么意义?能确保读到最新值?那么直接给Object[] elementData
加volatile
关键字保证可见性。
我也知道CopyOnWriteArrayList修改是每次重新建一个elementData
,但就算不是重新建,读时不加锁有什么问题?请赐教。
//以下是CopyOnWriteArrayList的代码:
@SuppressWarnings("unchecked")
private E get(Object[] a, int index) {
return (E) a[index];
}
/**
* {@inheritDoc}
*
* @throws IndexOutOfBoundsException {@inheritDoc}
*/
public E get(int index) {
return get(getArray(), index);
}
//以下是Vector的代码:
public synchronized E get(int index) {
if (index >= elementCount)
throw new ArrayIndexOutOfBoundsException(index);
return elementData(index);
}
Java の各スレッドには独自の作業メモリがあり、これはメイン メモリのコピーとして理解できます。 volatile を使用した場合、メイン メモリ上で即座に読み取れることのみが保証され、各スレッドの作業メモリとのアトミックな同期は保証されません
volatile を使用せず、書き込み操作に synchronized を追加するだけで読み取り操作には追加しない場合でも、最新の値を読み取れないという問題だけでなく、より大きな問題は同期です。スレッドのワーキングメモリとメインメモリの関係を次のような状況を想像してください:
整数 i = 0
スレッドは i + 1 を与えます
a がこの +1 操作を実行している間、スレッド b は i を自身の作業メモリに読み取ります。この時点では、読み取りはロックされていないため、i はまだ 0 です。
スレッド a が実行を完了し、i = 1 になり、メイン メモリに同期されます
b は i に対して +1 操作を実行します。a が書き込みロックを解放したため、b は書き込み可能ですが、自身の作業メモリ内の i は 0 であるため、+1