Java并发包(java.util.concurrent.atomic)提供了一系列原子操作类,这些类通过无锁算法实现了线程安全的操作,相比传统的锁机制具有更高的性能。原子类基于CAS(Compare-And-Swap)指令实现,是现代并发编程的重要基础。
原子类主要分类:
CAS是一种原子指令,包含三个操作数:
当且仅当V的值等于A时,处理器才会将V的值更新为B,否则不执行任何操作。无论哪种情况都会返回V的当前值。
Java通过Unsafe类提供CAS操作支持:
public final class Unsafe {
public final native boolean compareAndSwapInt(
Object o, long offset, int expected, int x);
public final native boolean compareAndSwapLong(
Object o, long offset, long expected, long x);
public final native boolean compareAndSwapObject(
Object o, long offset, Object expected, Object x);
}
ABA问题:值从A变为B又变回A,CAS会认为没变化
循环时间长开销大:CAS失败会自旋重试,消耗CPU
只能保证一个变量的原子操作
public class AtomicInteger extends Number {
private static final Unsafe unsafe = Unsafe.getUnsafe();
private static final long valueOffset;
static {
try {
valueOffset = unsafe.objectFieldOffset
(AtomicInteger.class.getDeclaredField("value"));
} catch (Exception ex) { throw new Error(ex); }
}
private volatile int value;
public final int getAndIncrement() {
return unsafe.getAndAddInt(this, valueOffset, 1);
}
// Unsafe中的实现
public final int getAndAddInt(Object o, long offset, int delta) {
int v;
do {
v = getIntVolatile(o, offset);
} while (!compareAndSwapInt(o, offset, v, v + delta));
return v;
}
}
设计思想:分段CAS,减少竞争
适用场景:高并发统计计数,不保证实时精确
public class Counter {
private final AtomicLong count = new AtomicLong(0);
public void increment() {
count.incrementAndGet();
}
public long getCount() {
return count.get();
}
}
public class Singleton {
private static final AtomicReference<Singleton> INSTANCE =
new AtomicReference<>();
private Singleton() {}
public static Singleton getInstance() {
for (;;) {
Singleton instance = INSTANCE.get();
if (instance != null) {
return instance;
}
instance = new Singleton();
if (INSTANCE.compareAndSet(null, instance)) {
return instance;
}
}
}
}
public class StatusManager {
private final AtomicBoolean status = new AtomicBoolean(false);
public void enable() {
status.set(true);
}
public boolean tryDisable() {
return status.compareAndSet(true, false);
}
}
问题描述:多个线程修改同一缓存行的不同变量,导致性能下降
解决方案:
public class PaddedAtomicLong extends AtomicLong {
public volatile long p1, p2, p3, p4, p5, p6 = 7L; // 填充
public PaddedAtomicLong(long initialValue) {
super(initialValue);
}
}
场景 | 推荐类 | 原因 |
---|---|---|
低竞争环境 | AtomicLong | 实现简单,开销小 |
高并发写入 | LongAdder | 分段减少竞争,吞吐量高 |
需要复杂累加操作 | LongAccumulator | 支持自定义累加函数 |
测试场景:100个线程,每个线程递增计数器100,000次
实现方式 | 耗时(ms) | 特点 |
---|---|---|
synchronized | 420 | 稳定但性能一般 |
ReentrantLock | 380 | 略优于synchronized |
AtomicInteger | 120 | 无锁,性能最好 |
LongAdder | 85 | 高并发下性能最优 |
结论:在适合的场景下,原子类性能显著优于锁机制
答案:不能。原子类适合简单原子操作,复杂同步仍需锁机制
原因:
建议:
Java原子操作类是基于CAS实现的高性能线程安全工具,理解其原理和适用场景对于编写高效并发程序至关重要。在实际开发中,应根据具体场景选择合适的原子类,并注意其内存语义和潜在问题。掌握原子类的使用技巧,能够在保证线程安全的同时获得接近于无锁的性能,是Java并发编程的高级技能之一。