学习是最好的投资!
是地址,准确的说是this对象的内存valueOffset的位置的值和expect比较.Unsafe详解
就如题主所说的,AtomicInteger源码是比较并替换Integer来实现线程安全性。而AtomicReference是相比于对象引用的比较并替换。这些都是原子类CAS实现。
AtomicInteger
Integer
AtomicReference
至于是不是比较地址,先拿题主所说的AtomicReference入手,知道下面的方法:
public final boolean compareAndSet(V expect, V update) { return unsafe.compareAndSwapObject(this, valueOffset, expect, update); }
底层实现在Unsafe类中,是一个native本地方法。Unsafe的CAS包括了三个操作数--需要读写的内存位置valueOffset,进行比较的值expected,拟定写入的新值update。当且仅当在内存位置V所存储的值等于比较的值A时,CAS才会通过原子方式用新值来更新内存地址的旧值。否则不进行任何操作。
Unsafe
native
valueOffset
expected
update
关键在于传入的valueOffset的作用,继续查看valueOffset的作用,继续查看AtomicReference源码:
static { try { valueOffset = unsafe.objectFieldOffset (AtomicReference.class.getDeclaredField("value")); } catch (Exception ex) { throw new Error(ex); } } private volatile V value;
这里,unsafe的objectFieldOffset方法是拿到对象的内存偏移量,即通过这里的比较即可判断是否同一个对象地址。
unsafe
objectFieldOffset
所以,得出结论:
Atomic原子类CAS操作比较的是内存偏移量,即内存地址。
是地址,准确的说是this对象的内存valueOffset的位置的值和expect比较.
Unsafe详解
就如题主所说的,
AtomicInteger
源码是比较并替换Integer
来实现线程安全性。而AtomicReference
是相比于对象引用的比较并替换。这些都是原子类CAS实现。至于是不是比较地址,先拿题主所说的
AtomicReference
入手,知道下面的方法:底层实现在
Unsafe
类中,是一个native
本地方法。Unsafe
的CAS包括了三个操作数--需要读写的内存位置valueOffset
,进行比较的值expected
,拟定写入的新值update
。当且仅当在内存位置V所存储的值等于比较的值A时,CAS才会通过原子方式用新值来更新内存地址的旧值。否则不进行任何操作。关键在于传入的
valueOffset
的作用,继续查看valueOffset
的作用,继续查看AtomicReference
源码:这里,
unsafe
的objectFieldOffset
方法是拿到对象的内存偏移量,即通过这里的比较即可判断是否同一个对象地址。所以,得出结论: