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);
}
Setiap thread dalam Java mempunyai memori kerja sendiri, yang boleh difahami sebagai salinan memori utama. Jika tidak menentu digunakan, ia hanya boleh menjamin bahawa ia boleh dibaca serta-merta pada memori utama, dan tidak dapat menjamin penyegerakan atom dengan memori kerja setiap utas
Walaupun anda tidak menggunakan volatile, tetapi hanya menambah disegerakkan pada operasi menulis, tetapi tidak pada operasi membaca, bukan sahaja masalah tidak dapat membaca nilai terkini, masalah yang lebih besar ialah penyegerakan ingatan kerja benang dan ingatan utama Bayangkan Situasi berikut:
Integer i = 0
benang memberikan i + 1
Semasa a melakukan operasi +1 ini, utas b membaca i ke dalam ingatan kerjanya sendiri Pada masa ini, i masih 0 kerana bacaan tidak dikunci
Benang a menyelesaikan pelaksanaan, i = 1, dan kemudian disegerakkan ke memori utama
b melakukan operasi +1 pada i Memandangkan a telah mengeluarkan kunci tulis, b boleh menulis, tetapi kerana i dalam ingatan kerjanya sendiri ialah 0, i masih 1 selepas +1