0

0

C++ atomic原子变量详解_C++无锁编程基础与线程安全计数器

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-11-27 15:26:02

|

824人浏览过

|

来源于php中文网

原创

atomic 原子变量是 C++11 提供的用于解决多线程竞争的机制,通过硬件级原子操作保证变量访问的完整性。它定义在 头文件中,支持如 int、bool 等基础类型的原子读写、自增、比较交换等操作,避免使用互斥锁实现线程安全。典型应用场景为计数器,例如多个线程对 std::atomic 类型变量进行自增,可确保结果正确。其核心操作包括 load、store、exchange 和 compare_exchange_weak/strong,并支持运算符重载。通过指定内存序(如 memory_order_relaxed、memory_order_seq_cst)可平衡性能与一致性。但 atomic 不适用于涉及多个变量或复杂逻辑的场景,此类情况仍需互斥锁。

c++ atomic原子变量详解_c++无锁编程基础与线程安全计数器

在多线程编程中,多个线程同时访问共享数据容易引发竞争条件(race condition),导致程序行为不可预测。C++ 提供了 atomic 类型来解决这一问题,它通过硬件级别的原子操作保证对变量的读、写或修改是不可分割的,从而实现线程安全而无需显式使用互斥锁(mutex)。

什么是 atomic 原子变量?

atomic 是 C++11 引入的一个模板类,定义在 头文件中。它可以包装一个基础类型(如 int、bool、指针等),并确保对该变量的操作是原子的。也就是说,当一个线程正在读取或修改该变量时,其他线程无法中断这个过程。

例如:

#include 
std::atomic counter(0); // 原子整型变量

此时对 counter 的自增、赋值等操作都是线程安全的。

立即学习C++免费学习笔记(深入)”;

为什么需要 atomic?

考虑一个简单的计数器场景:多个线程同时执行 counter++。表面上看是一条语句,实际上包含三个步骤:读取值、加1、写回内存。如果两个线程几乎同时读到相同的旧值,就会导致其中一个线程的更新被覆盖。

使用普通变量会出现漏计数的问题。而使用 std::atomic 后,整个“读-改-写”过程由 CPU 指令保证原子性,避免了数据竞争。

常见用法与操作

atomic 支持多种原子操作,常用的包括:

  • load():原子地读取当前值
  • store(val):原子地写入新值
  • exchange(val):设置新值,并返回旧值
  • compare_exchange_weak()compare_exchange_strong():比较并交换(CAS),用于实现无锁算法的核心机制
  • 支持运算符重载,如 ++、--、+= 等(仅适用于特定类型,如整型和指针)

示例:线程安全的计数器

PixVerse
PixVerse

PixVerse是一款强大的AI视频生成工具,可以轻松地将多种输入转化为令人惊叹的视频。

下载
#include 
#include 
#include 
#include 

std::atomic count(0);

void increment() {
    for (int i = 0; i < 1000; ++i) {
        count++; // 安全的原子自增
    }
}

int main() {
    std::vector threads;
    for (int i = 0; i < 10; ++i) {
        threads.emplace_back(increment);
    }

    for (auto& t : threads) {
        t.join();
    }

    std::cout << "Final count: " << count.load() << std::endl;
    return 0;
}

无论运行多少次,结果都会是 10000,说明 atomic 成功避免了竞态条件。

内存序(Memory Order)控制性能与可见性

atomic 操作可以指定内存顺序,影响编译器优化和 CPU 乱序执行的行为。默认使用 memory_order_seq_cst(顺序一致性),最安全但可能稍慢。

根据需求可选择更轻量的选项:

  • memory_order_relaxed:只保证原子性,不保证顺序,适合计数器类场景
  • memory_order_acquire / release:用于线程间同步,控制变量修改的可见顺序
  • memory_order_acq_rel:结合 acquire 和 release
  • memory_order_seq_cst:最强一致性模型

例如,一个高性能计数器可以这样写:

count.fetch_add(1, std::memory_order_relaxed);

这在不需要严格同步顺序时能提升性能。

限制与注意事项

并不是所有类型都能用于 atomic。标准库为 bool、整型、指针等提供了特化版本,但自定义类型需满足 trivially copyable 才能使用 atomic,且某些平台可能不支持复杂类型的原子操作。

另外,虽然 atomic 避免了锁的开销,但它并不能替代 mutex 在所有场景下的作用。比如涉及多个变量的复合逻辑,仍需使用锁来保证整体事务性。

基本上就这些。掌握 atomic 是学习 C++ 无锁编程的第一步,合理使用可以在保证线程安全的同时提升并发性能。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1465

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

316

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

538

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

192

2024.02.23

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

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

9

2026.01.16

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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