0

0

c++怎么使用std::atomic实现原子操作_c++原子操作与并发控制方法

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-10-27 23:14:01

|

743人浏览过

|

来源于php中文网

原创

std::atomic 是C++中用于实现线程安全原子操作的模板类,可避免数据竞争并减少锁开销。它支持整型、指针等可平凡复制类型,常用操作包括 load、store、fetch_add、compare_exchange_weak 等,适用于计数器、标志位等场景。通过指定 memory_order 可调节内存顺序以平衡性能与一致性,如 relaxed 用于高性能计数,acquire/release 用于同步,seq_cst 为默认强顺序。自定义类型需满足特定条件才能原子化,否则需借助锁。正确使用 std::atomic 能提升并发效率,但需注意类型限制与内存序选择。

c++怎么使用std::atomic实现原子操作_c++原子操作与并发控制方法

在C++中,std::atomic 是实现原子操作的核心工具,它能确保对共享变量的操作不会被多个线程同时修改而产生数据竞争。使用 std::atomic 可以避免使用互斥锁(mutex)带来的性能开销,适用于简单的共享状态同步场景,比如计数器、标志位等。

std::atomic 基本用法

std::atomic 是一个模板类,可以包装整型、指针等支持原子操作的类型。常用的基本类型如 intboollong 等都可以直接使用。

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

#include 
#include 
#include 

std::atomic counter(0);

void increment() {
    for (int i = 0; i < 1000; ++i) {
        counter.fetch_add(1, std::memory_order_relaxed);
    }
}

int main() {
    std::thread t1(increment);
    std::thread t2(increment);

    t1.join();
    t2.join();

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

上面代码中,两个线程同时对 counter 进行自增操作,由于使用了 std::atomic,整个操作是线程安全的,最终结果为2000。

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

常用的原子操作函数

std::atomic 提供了一系列成员函数用于执行原子读写和修改操作:

  • load():原子地读取当前值
  • store(val):原子地写入新值
  • fetch_add(val):原子加法,返回旧值
  • fetch_sub(val):原子减法
  • exchange(val):设置新值并返回旧值
  • compare_exchange_weak(expected, desired):比较并交换(CAS),常用于无锁编程

例如,使用 compare_exchange_weak 实现线程安全的单例或状态切换:

Img.Upscaler
Img.Upscaler

免费的AI图片放大工具

下载
std::atomic flag(false);

bool expected = false;
if (flag.compare_exchange_weak(expected, true)) {
    // 成功将 flag 从 false 改为 true
    std::cout << "Flag acquired!" << std::endl;
} else {
    std::cout << "Flag already set." << std::endl;
}

内存顺序(Memory Order)控制

原子操作的性能和可见性可以通过指定内存顺序来调节。C++ 提供了多种 std::memory_order 枚举值:

  • std::memory_order_relaxed:最宽松,只保证原子性,不保证顺序
  • std::memory_order_acquire:用于读操作,保证之后的读写不会被重排到此操作前
  • std::memory_order_release:用于写操作,保证之前的读写不会被重排到此操作后
  • std::memory_order_acq_rel:同时具有 acquire 和 release 语义
  • std::memory_order_seq_cst:默认选项,提供最严格的顺序一致性,性能开销最大

在不需要严格顺序的场景下,使用 relaxed 可提升性能。例如计数器:

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

而在需要同步多个变量时,通常使用 acquire/release 模式来建立 happens-before 关系。

适用类型与限制

std::atomic 并非支持所有类型。只有满足“可平凡复制(trivially copyable)”且硬件支持原子操作的类型才能使用。标准库对以下类型提供了特化:

  • 整型(int, long, bool 等)
  • 指针类型(T*)

对于自定义结构体,不能直接使用 std::atomic,除非该类型是字节可复制且编译器确认支持原子操作。否则需配合锁或其他机制实现原子性。

基本上就这些。合理使用 std::atomic 能有效减少锁的使用,提高并发效率,但要注意内存顺序的选择和类型限制,避免误用导致隐蔽的并发问题。

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

184

2025.07.04

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

184

2025.07.04

string转int
string转int

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

311

2023.08.02

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

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

510

2024.08.29

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

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

46

2025.08.29

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

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

177

2025.08.29

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

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

精品课程

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

共578课时 | 37.4万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 0.9万人学习

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

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