Java原子类通过CAS实现线程安全,依赖CPU硬件支持,采用乐观锁避免加锁开销,在低竞争下性能优于传统锁;ABA问题可通过AtomicStampedReference的版本戳解决;并发包还提供多种原子类如AtomicLong、AtomicReference及LongAdder等,适用于计数、状态标记、对象引用更新及高并发累加等场景。

Java中的原子类,比如
AtomicInteger
CAS操作是Java原子类实现线程安全的核心。它本质上是一种处理器指令,包含三个操作数:内存位置V(要操作的变量)、预期原值A(期望该位置V的值是A)、新值B(如果V等于A,则将V更新为B)。这个操作是原子的,意味着在多线程环境下,操作系统或CPU会保证这一整个“比较-更新”的步骤不会被其他线程中断。
在
AtomicInteger
getAndIncrement()
AtomicInteger
synchronized
Lock
CAS在某些特定场景下确实能展现出比传统锁机制更高的效率,这背后有几个关键原因。传统锁,无论是
synchronized
ReentrantLock
立即学习“Java免费学习笔记(深入)”;
而CAS则是一种“乐观锁”策略,它假设并发冲突不频繁。它直接尝试修改,如果发现数据被其他线程改动了,就重新尝试。这个过程是无锁的,意味着线程不需要在内核态和用户态之间频繁切换,也不需要被操作系统挂起和唤醒。当竞争不激烈时,CAS操作通常一次就能成功,避免了锁带来的所有开销。即使失败了,也只是自旋重试,这个自旋通常发生在用户态,消耗的是CPU时间片,但不会引起线程阻塞和上下文切换的重量级操作。因此,在低到中等程度的并发竞争下,CAS的这种“无锁”特性,能显著提升程序的吞吐量和响应速度。当然,高并发下,如果CAS频繁失败导致大量自旋,它也可能变得低效,甚至比锁更糟,因为它会一直占用CPU,造成“忙等”。
CAS操作确实存在一个潜在的陷阱,我们称之为“ABA”问题。简单来说,就是当一个变量V从A变为B,然后又变回了A。对于CAS操作来说,它会认为这个变量的值“没有变化”,因为它只比较当前值和预期值是否相等。但实际上,这个值在中间经历了一次或多次修改,可能导致一些依赖于“值未曾改变”的业务逻辑出现问题。
举个例子,你从银行取钱,账户余额是100元(A)。你发起一个CAS操作,期望将100元更新为50元。但在你的CAS操作执行前,另一个线程先存入了50元,余额变为150元(B),然后又取走了100元,余额再次变为50元(A)。此时你的CAS操作检查发现,当前余额确实是100元(A),与你期望的原值相同,于是成功将余额更新为50元。但问题是,这100元已经不是你最初看到的那个100元了,中间发生了两次操作,这可能在某些复杂的业务场景下导致逻辑错误。
为了解决ABA问题,Java并发包提供了
AtomicStampedReference
AtomicStampedReference
compareAndSet
AtomicMarkableReference
Java的
java.util.concurrent.atomic
AtomicLong
AtomicBoolean
AtomicInteger
long
boolean
AtomicLong
AtomicBoolean
AtomicReference<V>
AtomicReference
AtomicStampedReference<V>
AtomicMarkableReference<V>
AtomicStampedReference
AtomicMarkableReference
AtomicIntegerArray
AtomicLongArray
AtomicReferenceArray<E>
LongAdder
DoubleAdder
AtomicLong
AtomicLong
LongAdder
DoubleAdder
这些原子类是构建高效、无锁并发程序的重要基石,它们在底层利用了CAS指令,将并发控制的复杂性封装起来,让开发者能更专注于业务逻辑的实现。
以上就是Java中的原子类(如AtomicInteger)是如何利用CAS实现线程安全的?的详细内容,更多请关注php中文网其它相关文章!
                        
                        每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
                Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号