std::memory_order for std::atomic:<T>:wait
std::memory_order for std::atomic<T>::wait
根据cppreference,在C++20中,std::atomic<T>
中有wait
、notify_one
、notify_all
。看起来它们使std::atomic<T>
可用作蒲团。
我在问wait
为什么接受std::memory_order作为参数。由于我总是需要检查从等待中唤醒是否不是伪造的,我将在相应的加载中指定内存顺序:
std::atomic<bool> x;
while (x.load(std::memory_order_acquire) == false)
{
x.wait(false, std::memory_order_acquire);
}
或者我应该指定std::memory_order_relaxed
等待吗?是否存在wait
之后没有load
的情况?
同一对象上的所有原子操作都将始终具有单个顺序,并且单个线程内的顺序也包含在其中。也就是说,relaxed
不能对同一线程中同一对象上的原子操作进行重新排序。
relaxed
内存顺序是相对于除特定原子操作之外的其他内存操作的顺序。因此,relaxed
可以检查等待本身,但当您想实际读取由不包含在原子值中的发布线程写入的数据时,您需要更强的内存顺序来确保它们的可见性。因此,如果有必要,外循环应该使用更合适的条件来获得可见性。
我将这个问题发布到了GitHub上,并得到了一个响应,即std::atomic::wait旨在通过逻辑在futex上实现,特别是为了屏蔽虚假唤醒。
因此,cppreference.com在这方面是错误的:
这些函数被允许虚假解锁,即由于值更改或通知以外的原因返回。
虽然循环在我的例子中是多余的,但我应该只使用:
std::atomic<bool> x;
x.wait(false, std::memory_order_acquire);
相关文章:
- 使用std::multimap迭代器创建std::list
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 如何导出包含具有"std::unique_ptr"值的"std::map"属性的
- 从持续时间构造std::chrono::system_clock::time_point
- std::具有相同基类的类的变体
- std::向量与传递值的动态数组
- 使用std::vector的OpenCL矩阵乘法
- std::map<struct,struct>::find 找不到匹配项,但是如果我循环通过 begin() 到 end(),我在那里看到匹配项
- std::condition_variable::wait()如何评估给定的谓词
- 如何获取std::result_of函数的返回类型
- std::原子加载和存储都需要吗
- 将对象移动到std::shared_ptr
- std::vector<;uint8_t>;当C++11/14启用时,手动复制而不是调用memcpy
- 是std :: set&lt; std :: future&gt;不可能存在
- 在修改列表后,std :: list&lt; t&gt; :: end()的值是否会更改
- 使用 std::vector<boost::shared_ptr<Base_Class>> 或 boost::p tr_vector 的性能注意事项是什么<Base>
- std::map<std::set, double> AND std:<long>map< std::p air<long, long>, double>
- 如何获取std::vector<DMatch>