对 std::atomic 变量的更改(读/写)如何在线程之间传播
How do changes (read/writes) to std::atomic variables propagate across threads
我最近问了这个问题,我需要使用内存屏障来保护共享资源
对于这个问题,我得到了一个非常有趣的答案,它使用了这个假设:
Changes to std::atomic variables are guaranteed to propagate across threads.
为什么会这样?怎么做?此行为如何适应 MESI 协议?
它们实际上不必传播,缓存一致性模型(MESI或更高级的模型)为您提供了内存连贯行为的保证,几乎就像它是平面的并且不存在缓存副本一样。顺序一致性增加了对系统中所有代理相同观察顺序的保证(注意 - 大多数 CPU 不能仅通过硬件提供顺序一致性)。
如果线程执行内存写入(甚至不是原子的),则运行该行的核心将获取该行并获得对该行的所有权。写入完成后,任何尝试观察该行的线程都可以保证看到更新的值,即使该行仍驻留在修改核心中 - 通常这是通过窥探核心并从中获取该行作为响应来实现的。缓存一致性协议将保证,如果这样的修改存在于某个内核中 - 任何其他寻找该行的核心最终都会看到它。为此,CPU 我使用侦听过滤器、目录管理(通常用于跨套接字一致性)或其他方法。
现在,你要问为什么原子很重要?原因有二。首先 - 以上所有内容仅适用于变量驻留在内存中而不是寄存器中的情况。这是一个编译器决策,因此正确的类型会告诉它这样做。其他范式(如 open-MP 或 POSIX 线程)有其他方法告诉编译器变量需要通过内存共享。其次,现代内核无序执行操作,我们不希望任何其他操作传递写入并公开过时数据。std::atomic告诉编译器强制实施最强的内存排序(通过使用显式屏蔽或锁定 - 检查生成的汇编代码),这意味着来自所有线程的所有内存操作将具有相同的全局排序。如果你不这样做,可能会发生奇怪的事情,比如核心 A 和核心 B 不同意 2 次写入同一位置的顺序(这意味着他们可能会在其中看到不同的最终值)。
最后,当然,是实际的原子性 - 如果你的数据类型没有保证原子性,或者它没有正确对齐 - 这也将为你解决这个问题(否则一致性问题会加剧 - 想想一些线程试图改变在 2 个缓存行之间拆分的值,不同的内核看到部分值)
- 从不同线程使用int64的不同字节安全吗
- 删除一个线程上有数百万个字符串的大型哈希映射会影响另一个线程的性能
- 在C++中使用cURL和多线程
- 为什么我的C#代码在调用回C++COM直到Task时会暂停.等待/线程.加入
- 在cuda线程之间共享大量常量数据
- 如何将元素添加到数组的线程安全函数?
- 线程,如果else语句,都是错误的上下文切换后,会发生什么
- C++Boost Asio Pool线程,带有lambda函数和传递引用变量
- Qt C++静态thread_local QNetworkAccessManager是线程应用程序的好选择吗
- 异常属于C++中的线程还是进程
- C++中的线程安全删除
- C++使用params创建线程函数会导致转换错误
- 类与私有变量的其他类之间的线程安全性
- CoInitialize()在单独的线程上崩溃而不返回
- c++中的线程池
- 线程之间的布尔停止信号
- 为什么std::async使用同一个线程运行函数
- 用于矢量处理的多个线程
- 异常不会使用boost :: future/boost :: Promise跨线程正确传播
- 向c++ 11线程传播信号(SIGINT)