0

0

说一下 atomic 的原理?

煙雲

煙雲

发布时间:2026-01-18 01:59:31

|

966人浏览过

|

来源于php中文网

原创

Atomic操作是不可分割的,依赖CAS等硬件指令保证多线程下数据一致性,避免竞态条件;Java中AtomicInteger通过CAS实现原子增减,但高并发时可能因自旋导致性能下降;ABA问题可通过AtomicStampedReference加版本号解决;除CAS外,锁、事务内存、HTM也可实现Atomic操作,但CAS仍是最常用方式。

说一下 atomic 的原理?

Atomic操作,简单来说,就是不可分割的操作。在并发编程里,这意味着一个操作要么完全执行成功,要么完全不执行,不存在中间状态。这保证了多线程环境下数据的一致性,避免出现竞态条件。

解决方案 Atomic操作的原理依赖于底层硬件和操作系统的支持。通常,CPU提供了一些特殊的指令,比如Compare-and-Swap (CAS) 指令,可以原子地比较内存中的值与预期值,如果相等则更新为新值。操作系统则会提供相应的API,封装这些硬件指令,并处理更复杂的并发场景。

CAS操作是Atomic操作的核心。它包含三个操作数:内存地址V,预期值A,和新值B。CAS操作会比较内存地址V的值是否等于预期值A,如果相等,那么将内存地址V的值更新为新值B,否则不进行任何操作。整个比较和更新的过程是一个原子操作。

举个例子,假设我们要实现一个原子计数器。我们可以使用一个AtomicInteger类,它内部维护一个int类型的变量,并提供incrementAndGet()方法来原子地增加计数器的值。这个方法内部就是使用CAS操作来实现的。

import java.util.concurrent.atomic.AtomicInteger;

public class AtomicCounter {

    private AtomicInteger counter = new AtomicInteger(0);

    public int incrementAndGet() {
        return counter.incrementAndGet();
    }

    public int get() {
        return counter.get();
    }

    public static void main(String[] args) throws InterruptedException {
        AtomicCounter counter = new AtomicCounter();

        // 模拟多线程并发增加计数器
        Thread[] threads = new Thread[10];
        for (int i = 0; i < 10; i++) {
            threads[i] = new Thread(() -> {
                for (int j = 0; j < 1000; j++) {
                    counter.incrementAndGet();
                }
            });
            threads[i].start();
        }

        // 等待所有线程执行完成
        for (int i = 0; i < 10; i++) {
            threads[i].join();
        }

        System.out.println("Counter value: " + counter.get()); // 预期结果是 10000
    }
}

在这个例子中,即使有多个线程同时调用incrementAndGet()方法,由于AtomicInteger使用了CAS操作,所以计数器的值仍然能够正确地增加。

Atomic操作一定比锁效率高吗?

不一定。虽然Atomic操作避免了锁的开销,但CAS操作在高并发场景下可能会出现大量的自旋重试,导致CPU资源的浪费。如果竞争非常激烈,锁可能会更加高效。此外,锁还可以提供一些额外的功能,比如公平性,而Atomic操作则无法保证。选择使用Atomic操作还是锁,需要根据具体的应用场景进行权衡。在低并发、读多写少的场景下,Atomic操作通常是更好的选择。在高并发、写多读少的场景下,可能需要考虑使用锁或其他并发控制机制。

ABA问题是什么,Atomic如何解决?

ABA问题是CAS操作中一个常见的问题。假设一个变量V的初始值是A,线程1将V的值从A修改为B,然后又修改回A。此时,线程2使用CAS操作,期望将V的值从A修改为C。由于V的值仍然是A,CAS操作会成功,但实际上V的值已经经历了A->B->A的变化。

Atomic包提供了一个AtomicStampedReference类,可以解决ABA问题。AtomicStampedReference类维护了一个值和一个版本号(stamp),每次修改值的同时,也需要修改版本号。在进行CAS操作时,需要同时比较值和版本号。这样,即使值相同,但版本号不同,CAS操作也会失败。

import java.util.concurrent.atomic.AtomicStampedReference;

public class ABADemo {

    private static AtomicStampedReference atomicStampedRef =
            new AtomicStampedReference<>(1, 0);

    public static void main(String[] args) throws InterruptedException {

        Thread thread1 = new Thread(() -> {
            int stamp = atomicStampedRef.getStamp();
            System.out.println("Thread1 initial stamp: " + stamp);

            try {
                Thread.sleep(1000); // 模拟线程1的延迟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            boolean success = atomicStampedRef.compareAndSet(1, 2, stamp, stamp + 1);
            System.out.println("Thread1 CAS result: " + success + ", current stamp: " + atomicStampedRef.getStamp());

            success = atomicStampedRef.compareAndSet(2, 1, atomicStampedRef.getStamp(), atomicStampedRef.getStamp() + 1);
            System.out.println("Thread1 CAS result: " + success + ", current stamp: " + atomicStampedRef.getStamp());
        });

        Thread thread2 = new Thread(() -> {
            int stamp = atomicStampedRef.getStamp();
            System.out.println("Thread2 initial stamp: " + stamp);

            try {
                Thread.sleep(3000); // 模拟线程2的延迟
            } catch (InterruptedException e) {
                e.printStackTrace();
            }

            boolean success = atomicStampedRef.compareAndSet(1, 3, stamp, stamp + 1);
            System.out.println("Thread2 CAS result: " + success + ", current stamp: " + atomicStampedRef.getStamp());
        });

        thread1.start();
        thread2.start();

        thread1.join();
        thread2.join();

        System.out.println("Final value: " + atomicStampedRef.getReference());
        System.out.println("Final stamp: " + atomicStampedRef.getStamp());
    }
}

在这个例子中,线程1首先将值从1改为2,然后再改回1。线程2尝试将值从1改为3,由于线程1修改了版本号,所以线程2的CAS操作会失败,从而避免了ABA问题。

除了CAS,还有其他实现Atomic的机制吗?

除了CAS,还有一些其他的机制可以实现Atomic操作,例如:

  • 锁机制: 虽然Atomic操作通常被认为是无锁的,但在某些情况下,底层实现仍然可能使用锁来保证原子性。例如,当CAS操作失败时,可能会使用自旋锁或互斥锁来进行重试。
  • 事务内存: 事务内存是一种新兴的并发编程技术,它允许将多个操作组合成一个原子事务。如果事务执行过程中发生冲突,则会回滚事务,并重新执行。
  • 硬件事务内存 (HTM): 某些CPU提供了硬件级别的事务内存支持,可以更加高效地实现Atomic操作。例如,Intel的Transactional Synchronization Extensions (TSX) 指令集。

这些机制各有优缺点,选择使用哪种机制取决于具体的应用场景和硬件平台。通常情况下,CAS操作是实现Atomic操作最常用的方法,因为它具有较高的性能和较低的开销。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

835

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

740

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

736

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

43

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.9万人学习

Java 教程
Java 教程

共578课时 | 47.1万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号