简介
为了实现对变量的原子性更新,java 提供了工具类:AtomicReferenceFieldUpdater。
该类可实现对变量的 CAS 操作,填补以前只能对基本类型的原子性更新(例如AtomicInteger)。
关于 AtomicReferenceFieldUpdater 的解释:
一个基于反射的工具,实现原子性更新一个类实例中 volatile 修饰的字段。这个类被用于在原子性结构体中,一个类的几个变量希望被不同线程进行原子性更新。
这个类并不像其他原子性工具类(AtomicLong),不能保证对于字段的所有更新操作都是线程安全的,只能保证使用AtomicReferenceFieldUpdater工具类更新的操作是原子性的。
用例
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;
import org.junit.Test;
public class FieldUpdateTest {
protected volatile Data data;
private static final AtomicReferenceFieldUpdater<FieldUpdateTest, Data> updater =
AtomicReferenceFieldUpdater.newUpdater(FieldUpdateTest.class, Data.class, "data");
@Test
public void testAtomicReferenceFieldUpdater() {
Data data = new Data();
data.value = "value";
System.out.println("原始值:" + data.value);
System.out.println();
updater.set(this, data);
data = updater.get(this);
System.out.println("第一次赋值:" + data.value);
System.out.println();
data = updater.getAndUpdate(this, (d) -> {
System.out.println("更新前:" + d.value);
Data newData = new Data();
newData.value = "update value";
System.out.println("更新后:" + newData.value);
return newData;
});
System.out.println("获取后更新:" + data.value);
System.out.println();
data = updater.updateAndGet(this, (d) -> {
System.out.println("更新前:" + d.value);
Data newData = new Data();
newData.value = "update value 2";
System.out.println("更新后:" + newData.value);
return newData;
});
System.out.println("更新后获取:" + data.value);
System.out.println();
Data expect = data;
Data update = new Data();
update.value = "update value 3";
// 成功被更新
boolean flag = updater.compareAndSet(this, expect, update);
System.out.println("更新结果:" + flag);
flag = updater.compareAndSet(this, expect, update);
// 更新失败
System.out.println("更新结果:" + flag);
}
class Data {
String value;
}
}
输出结果:
原始值:value
第一次赋值:value
更新前:value
更新后:update value
获取后更新:value
更新前:update value
更新后:update value 2
更新后获取:update value 2
更新结果:true
更新结果:false
其他
被 volatile 修饰的字段才能使用AtomicReferenceFieldUpdater。