atomicinteger底层靠cas而非synchronized,通过unsafe.compareandswapint实现硬件级原子操作;incrementandget()返回新值,getandincrement()返回旧值;++非原子,易丢更新;compareandset需配合循环重试防失败。

AtomicInteger底层靠的是CAS还是synchronized
AtomicInteger不依赖 synchronized,核心是CPU级别的CAS(Compare-And-Swap)指令,由 Unsafe.compareAndSwapInt 实现。JVM会将它编译为对应平台的原子汇编指令(如x86的 cmpxchg),无需加锁,也没有线程阻塞开销。
注意:CAS不是万能的——存在ABA问题,但 AtomicInteger 不处理这个(因为它只关心数值是否变化,不关心“变过几次”;若需ABA防护,得用 AtomicStampedReference)。
incrementAndGet() 和 getAndIncrement() 的行为差异
两者都原子地加1,但返回值不同:
-
incrementAndGet()返回**更新后的新值**(先加再取) -
getAndIncrement()返回**更新前的旧值**(先取再加)
例如初始值为5:
立即学习“Java免费学习笔记(深入)”;
AtomicInteger ai = new AtomicInteger(5); int a = ai.incrementAndGet(); // a == 6,ai.get() == 6 int b = ai.getAndIncrement(); // b == 6,ai.get() == 7
选哪个取决于业务逻辑对“序号”或“计数器”的语义要求——比如生成唯一ID通常用 incrementAndGet(),而日志计数器可能更倾向 getAndIncrement()。
为什么不能直接用 ++ 操作符替代 AtomicInteger
++ 是三步操作:读内存 → 计算+1 → 写回内存。在多线程下,两个线程可能同时读到相同旧值,各自+1后再写回,导致只加了一次(丢失一次更新)。
而 AtomicInteger.addAndGet(1) 或 incrementAndGet() 把这三步打包成一个不可分割的硬件操作,失败时自动重试(自旋),保证结果正确。
常见误用场景:
- 把
AtomicInteger当普通 int 传参后做++:参数传递的是副本,原对象不受影响 - 在循环里反复调用
get()再手动set():这已脱离原子性保障
compareAndSet() 的典型使用模式和陷阱
compareAndSet(expectedValue, newValue) 是最灵活的原子操作,常用于实现无锁算法或条件更新。
典型用法示例(实现一个简单的“首次初始化”):
AtomicInteger flag = new AtomicInteger(0);
// 只有第一次调用会成功,返回 true
if (flag.compareAndSet(0, 1)) {
initialize();
}
容易踩的坑:
- 预期值写错:比如用
flag.get()的瞬时快照作为expectedValue,但该值可能已被其他线程改过,导致CAS失败——这不是bug,是设计使然,需要配合循环重试(即自旋) - 把
compareAndSet当作“乐观锁”却忽略重试逻辑,结果失败后静默跳过,造成逻辑遗漏 - 在高竞争场景下,频繁CAS失败会带来明显CPU自旋开销,此时需评估是否改用
ReentrantLock等显式锁
真正难的不是调用API,而是判断什么时候该用CAS、什么时候该让出CPU、以及如何避免ABA在业务语义中引发歧义。










