0

0

C++如何使用std::atomic_C++原子操作与多线程安全实践

尼克

尼克

发布时间:2025-11-20 14:05:02

|

202人浏览过

|

来源于php中文网

原创

std::atomic是C++11引入的模板类,用于保证对基本类型的读写操作具有原子性,避免多线程下的数据竞争。它通过提供load、store、fetch_add、exchange和compare_exchange_weak/strong等原子操作,实现无锁并发控制。相比互斥锁,std::atomic性能更高,适用于计数器、标志位同步等场景。配合内存序(如memory_order_relaxed、acquire、release、seq_cst)可精细控制操作顺序与可见性。例如用fetch_add可安全递增共享计数器,用release/acquire确保跨线程内存访问顺序,用CAS实现无锁栈或队列。正确使用需理解其内存模型,防止因误配内存序导致隐蔽错误。

c++如何使用std::atomic_c++原子操作与多线程安全实践

在C++多线程编程中,数据竞争是常见问题。当多个线程同时读写同一变量时,如果没有同步机制,程序行为将不可预测。std::atomic 提供了一种轻量级、高效的解决方案,确保对特定类型的操作是原子的,从而避免使用互斥锁(mutex)带来的性能开销。

什么是 std::atomic

std::atomic 是 C++11 引入的一个模板类,用于封装基本数据类型(如 int、bool、指针等),保证对其的操作是原子的。这意味着在多线程环境下,读取或修改该变量不会出现中间状态,也不会引发数据竞争。

常见的原子类型包括:

  • std::atomic
  • std::atomic
  • std::atomic
  • std::atomic(原子指针)

例如,定义一个原子整数:

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

std::atomic counter{0};

原子操作的基本用法

std::atomic 支持多种操作,最常用的是 load、store、fetch_add、exchange 和 compare_exchange_weak/strong。

示例:多线程递增计数器

#include
#include
#include

std::atomic count{0};

void increment() {
for (int i = 0; i count.fetch_add(1, std::memory_order_relaxed);
}
}

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

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

// 结果一定是 10000
return 0;
}

这里使用 fetch_add 原子地增加计数器值,即使多个线程同时调用,最终结果也正确。

ChartGen
ChartGen

AI快速生成专业数据图表

下载

内存序(Memory Order)的选择

std::atomic 操作可以指定内存顺序,控制操作的可见性和同步行为。常用的有:

  • std::memory_order_relaxed:只保证原子性,不保证顺序。适合计数器等无需同步场景。
  • std::memory_order_acquire:用于读操作,确保后续读写不会被重排到该操作之前。
  • std::memory_order_release:用于写操作,确保前面的读写不会被重排到该操作之后。
  • std::memory_order_acq_rel:同时具备 acquire 和 release 语义。
  • std::memory_order_seq_cst:默认选项,提供最严格的顺序一致性,性能稍低。

例如,实现自旋锁或标志位同步时,可使用 acquire/release 来减少开销:

std::atomic ready{false};
int data = 0;

// 线程1:
data = 42;
ready.store(true, std::memory_order_release); // 保证 data 写入在 store 前完成

// 线程2:
while (!ready.load(std::memory_order_acquire)) {
// 自旋等待
}
// 此时能安全读取 data
assert(data == 42);

compare-and-swap 与无锁编程

compare_exchange_weak 和 compare_exchange_strong 实现了 CAS(Compare and Swap)操作,是构建无锁数据结构的基础。

典型用法:

std::atomic value{0};
int expected = value.load();
do {
int desired = expected + 1;
} while (!value.compare_exchange_weak(expected, desired));

这段代码尝试将 value 原子地加 1。如果期间被其他线程修改,expected 会更新为当前值,循环继续尝试,直到成功。

CAS 常用于实现无锁、队列等结构,避免锁竞争,提升并发性能。

基本上就这些。std::atomic 是多线程编程中的重要工具,合理使用能显著提高程序效率和安全性。关键是理解原子操作的语义和内存模型,避免误用导致隐蔽 bug。

相关专题

更多
数据类型有哪几种
数据类型有哪几种

数据类型有整型、浮点型、字符型、字符串型、布尔型、数组、结构体和枚举等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

301

2023.10.31

php数据类型
php数据类型

本专题整合了php数据类型相关内容,阅读专题下面的文章了解更多详细内容。

222

2025.10.31

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

197

2023.11.20

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

85

2023.09.25

string转int
string转int

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

315

2023.08.02

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

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

537

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

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

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

精品课程

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

共58课时 | 3.7万人学习

Pandas 教程
Pandas 教程

共15课时 | 0.9万人学习

ASP 教程
ASP 教程

共34课时 | 3.6万人学习

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

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