C++中std::atomic_ref怎么原子化普通变量

来源:语言推理作者:沙月恵奈‌头衔:网络博主
导读:本期聚焦于小伙伴创作的《C++中std::atomic_ref怎么原子化普通变量》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《C++中std::atomic_ref怎么原子化普通变量》有用,将其分享出去将是对创作者最好的鼓励。

在C++多线程编程中,原子操作是避免数据竞争的重要手段,以往我们通常使用std::atomic模板来定义原子变量,但这种方式要求变量本身必须是std::atomic类型,在很多已经定义好的普通变量场景中无法直接适用。C++20引入的std::atomic_ref解决了这个问题,它可以通过包装普通变量的引用,让该变量临时具备原子操作的特性,不需要修改变量本身的类型定义。

std::atomic_ref的基本使用

std::atomic_ref定义在<atomic>头文件中,它的模板参数就是要包装的变量类型,构造时传入普通变量的引用即可,之后通过这个atomic_ref对象调用原子操作接口,就能保证对原变量的操作是原子的。

下面是一个简单的使用示例,展示如何用std::atomic_ref原子化一个普通的int变量:

#include <atomic>
#include <iostream>
#include <thread>
#include <vector>

// 普通全局变量,不是std::atomic类型
int normal_counter = 0;

void increment_task(std::atomic_ref<int> ref) {
    for (int i = 0; i < 1000; ++i) {
        // 原子加1操作
        ref.fetch_add(1, std::memory_order_relaxed);
    }
}

int main() {
    // 构造atomic_ref包装普通变量
    std::atomic_ref<int> counter_ref(normal_counter);
    std::vector<std::thread> threads;
    // 启动10个线程同时操作普通变量
    for (int i = 0; i < 10; ++i) {
        threads.emplace_back(increment_task, counter_ref);
    }
    // 等待所有线程执行完成
    for (auto& t : threads) {
        t.join();
    }
    std::cout << "最终计数结果: " << normal_counter << std::endl; // 输出10000
    return 0;
}

std::atomic_ref的适用场景

std::atomic_ref并不是要完全替代std::atomic,它更适合以下场景:

  • 需要原子操作已经定义好的普通变量,无法修改其原有类型定义,比如第三方库暴露的普通变量接口。
  • 只有部分代码段需要对普通变量做原子操作,其他场景仍希望以普通方式访问该变量,避免std::atomic带来的额外开销。
  • 处理数组、结构体等聚合类型的场景,std::atomic不支持直接对聚合类型做原子操作,而std::atomic_ref可以包装聚合类型的引用,对单个成员或者整个对象做原子操作。

使用std::atomic_ref的注意事项

使用std::atomic_ref时需要注意以下几点,避免出现未定义行为:

1. 变量生命周期要求

std::atomic_ref只是普通变量的引用包装,并不管理原变量的生命周期,必须保证atomic_ref对象存活期间,原变量的生命周期没有结束,否则会引发悬垂引用问题。

2. 内存序参数的选择

和std::atomic一样,std::atomic_ref的所有原子操作都支持内存序参数,需要根据实际场景选择合适的内存序,默认使用std::memory_order_seq_cst,对性能要求高的场景可以选择更宽松的内存序。

3. 普通访问的并发问题

std::atomic_ref只能保证通过自身进行的操作是原子的,如果同时存在其他代码直接对原变量进行非原子访问,仍然会产生数据竞争,因此使用时要确保同一时间要么所有访问都通过同一个atomic_ref,要么其他访问都是只读的。

4. 可复制性限制

std::atomic_ref的模板参数必须是可原子操作的对象类型,比如普通的算术类型、指针类型等,不支持拷贝构造和拷贝赋值的类型无法作为std::atomic_ref的模板参数。

std::atomic_ref与std::atomic的对比

我们可以通过下面的表格更清晰地看到两者的区别:

对比项std::atomicstd::atomic_ref
变量类型要求变量本身必须是std::atomic类型变量可以是普通类型,只需传入引用
生命周期管理管理自身变量的生命周期不管理原变量生命周期,仅包装引用
适用场景长期需要原子操作的变量临时需要原子操作的普通变量
性能开销变量本身可能有额外的标志位开销无额外变量开销,仅包装引用

总的来说,std::atomic_ref是C++20给多线程开发带来的实用特性,它填补了普通变量原子化操作的空白,让开发者在不需要修改原有变量定义的前提下,就能安全地实现并发场景下的原子操作,在使用时只要注意生命周期和并发访问的规范,就能很好地发挥它的作用。

std::atomic_refC++20多线程并发原子操作修改时间:2026-06-23 22:39:58

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。