news 2026/6/9 23:10:14

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/6/9 21:20:17

什么是LLDP

文章目录为什么需要LLDPLLDP应用场景有哪些LLDP报文格式LLDP是如何工作的LLDP&#xff08;Link Layer Discovery Protocol&#xff09;是IEEE 802.1ab中定义的链路层发现协议。LLDP是一种标准的二层发现方式&#xff0c;可以将本端设备的管理地址、设备标识、接口标识等信息组织…

作者头像 李华
网站建设 2026/6/9 21:19:18

华为OD技术面真题 - 计算机网络 - 1

文章目录计算机网络体系计算机网络为什么要分层应用进程的数据在各层之间传递过程端口、IP地址和MAC地址分别的作用说说不同层经典网络协议计算机网络体系 计算机网络体系结构标准主要分为三种: OSI体系结构:概念清楚&#xff0c;理论也比较完整&#xff0c;但是它既复杂又不…

作者头像 李华
网站建设 2026/6/4 22:00:28

java进阶--多线程学习

java进阶–多线程学习 java进阶–多线程学习&#xff08;1&#xff09; java进阶–多线程学习&#xff08;1&#xff09; 1.并行与并发的概念 并发是指一个处理器同时处理多个任务。 并行是指多个处理器或者是多核的处理器同时处理多个不同的任务。 并发是逻辑上的同时发生&…

作者头像 李华
网站建设 2026/6/9 22:31:22

大数据时代 RabbitMQ 对数据质量的保障

大数据时代 RabbitMQ 对数据质量的保障关键词&#xff1a;大数据时代、RabbitMQ、数据质量保障、消息队列、可靠性传输摘要&#xff1a;在大数据时代&#xff0c;数据质量对于企业的决策和业务发展至关重要。RabbitMQ 作为一款广泛使用的消息队列中间件&#xff0c;在保障数据质…

作者头像 李华
网站建设 2026/6/5 10:52:07

AI(人工智能)是模拟人类智能行为的技术,如学习、推理、识别

AI&#xff08;人工智能&#xff09;是模拟人类智能行为的技术&#xff0c;如学习、推理、识别等。大模型通常指参数量巨大的深度学习模型&#xff08;如GPT、BERT&#xff09;&#xff0c;依赖海量数据和算力进行训练&#xff0c;在自然语言处理、图像生成等领域表现卓越。前端…

作者头像 李华