news 2026/4/16 10:18:12

C++内存序

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C++内存序

在 C++ 中,内存序(Memory Order)是多线程编程中原子操作的重要概念,它用于控制原子操作的内存同步行为。C++11 引入了<atomic>头文件,提供了内存序来控制多线程环境下的内存访问顺序。

内存序的作用

内存序主要解决两个问题:

  1. 可见性:一个线程对共享数据的修改何时对其他线程可见

  2. 顺序性:操作指令的执行顺序如何被其他线程观察

六种内存序

1.memory_order_relaxed

最宽松的顺序约束,只保证原子性,不保证顺序。

cpp

std::atomic<int> x(0); x.store(1, std::memory_order_relaxed); // 不保证其他线程立即看到这个值

2.memory_order_consume

依赖于该原子操作的后续操作不会被重排序到该操作之前(依赖关系)。

cpp

std::atomic<int*> ptr; int data; // 线程1 data = 42; ptr.store(&data, std::memory_order_consume); // 线程2 int* p = ptr.load(std::memory_order_consume); if (p != nullptr) { // 保证能看到 data = 42 int val = *p; }

3.memory_order_acquire

用于读操作,保证该操作之后的所有读写不会被重排序到该操作之前。

cpp

std::atomic<bool> flag(false); int data = 0; // 线程1 data = 42; flag.store(true, std::memory_order_release); // 线程2 while (!flag.load(std::memory_order_acquire)); // 这里保证能看到 data = 42

4.memory_order_release

用于写操作,保证该操作之前的所有读写不会被重排序到该操作之后。

cpp

// 与上面 acquire 配合使用

5.memory_order_acq_rel

同时包含 acquire 和 release 语义,用于读-修改-写操作。

cpp

std::atomic<int> counter(0); counter.fetch_add(1, std::memory_order_acq_rel);

6.memory_order_seq_cst

最严格的顺序约束(默认),保证所有线程看到相同的操作顺序。

cpp

std::atomic<int> x(0); x.store(1); // 默认使用 memory_order_seq_cst

典型使用模式

1.Release-Acquire 同步

cpp

std::atomic<bool> ready(false); int data = 0; // 线程1(生产者) data = 42; ready.store(true, std::memory_order_release); // 线程2(消费者) while (!ready.load(std::memory_order_acquire)); // 这里保证能看到 data = 42

2.Release-Consume 同步

cpp

std::atomic<int*> ptr(nullptr); int value; // 线程1 value = 100; ptr.store(&value, std::memory_order_release); // 线程2 int* p = ptr.load(std::memory_order_consume); if (p != nullptr) { // 保证能看到 p 指向的数据 int v = *p; // v = 100 }

3.自旋锁实现

cpp

class SpinLock { std::atomic_flag flag = ATOMIC_FLAG_INIT; public: void lock() { while (flag.test_and_set(std::memory_order_acquire)); } void unlock() { flag.clear(std::memory_order_release); } };

性能考虑

  • relaxed:性能最好,但需要谨慎使用

  • seq_cst:性能最差,但最容易理解

  • acquire/release:在性能和正确性之间取得平衡

实用建议

  1. 优先使用默认的 seq_cst,除非有性能瓶颈

  2. 理解 happens-before 关系后再使用宽松内存序

  3. 测试多线程代码,内存序错误很难调试

  4. 使用现成的同步原语(如 mutex, condition_variable)通常更安全

示例:无锁计数器

cpp

#include <atomic> #include <thread> #include <iostream> class Counter { std::atomic<int> count{0}; public: void increment() { count.fetch_add(1, std::memory_order_relaxed); } int get() const { return count.load(std::memory_order_acquire); } }; int main() { Counter counter; std::thread t1([&]() { for (int i = 0; i < 1000000; ++i) { counter.increment(); } }); std::thread t2([&]() { for (int i = 0; i < 1000000; ++i) { counter.increment(); } }); t1.join(); t2.join(); std::cout << "Count: " << counter.get() << std::endl; return 0; }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 10:17:56

深度测评2026最新!9款AI论文软件评测:本科生毕业论文全场景推荐

深度测评2026最新&#xff01;9款AI论文软件评测&#xff1a;本科生毕业论文全场景推荐 2026年AI论文工具测评&#xff1a;为何值得一看&#xff1f; 随着人工智能技术的不断进步&#xff0c;AI论文辅助工具在学术领域的应用日益广泛。对于本科生而言&#xff0c;撰写毕业论文不…

作者头像 李华
网站建设 2026/4/9 3:54:13

新手进阶Python:给办公助手加定时功能,自动处理任务不用盯

大家好&#xff01;我是CSDN的Python新手博主&#xff5e; 上一篇我们整合了Excel、Word、PDF、OCR的全能办公助手&#xff0c;很多小伙伴反馈“如果能定时自动运行就好了&#xff0c;比如每天早上8点自动处理Excel数据、生成报表并发送给领导&#xff0c;不用我手动点”。今天…

作者头像 李华
网站建设 2026/4/16 1:41:19

linux redis简单操作

linux redis简单操作 1、centOS 系统安装redis https://blog.csdn.net/weixin_42835409/article/details/119562074 2、安装完成后终端输入redis-cli 报 redis-cli -bash: redis-cli: command not found 问题解决 进入到redis文件下src/bin目录下 将redis-cli 复制到 /usr/loc…

作者头像 李华
网站建设 2026/4/15 13:32:41

为什么“缓存“能提高系统性能?——从 CPU 缓存到分布式缓存

&#x1f4a8; 为什么"缓存"能提高系统性能&#xff1f;——从 CPU 缓存到分布式缓存 ⚡大家好&#xff0c;我是无限大&#xff0c;欢迎收看十万个为什么系列文章 希望今天的内容能对大家有所帮助想象一下&#xff1a;你去图书馆借一本书&#xff0c;每次都要从书架上…

作者头像 李华
网站建设 2026/4/15 13:50:23

java进阶知识点

java回收机制 浅谈java中的反射 依赖注入的简单理解 通过接口的引用和构造方法的表达&#xff0c;将一些事情整好了反过来传给需要用到的地方~ 这样做得好处&#xff1a;做到了单一职责&#xff0c;并且提高了复用性&#xff0c;解耦了之后&#xff0c;任你如何实现&#xf…

作者头像 李华