- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
在回答 this question关于 OP 情况的另一个问题出现了,我不确定:这主要是处理器架构问题,但也有一个关于 C++ 11 内存模型的链式问题。
基本上,由于以下代码(为简单起见稍作修改),OP 的代码在更高的优化级别无限循环:
while (true) {
uint8_t ov = bits_; // bits_ is some "uint8_t" non-local variable
if (ov & MASK) {
continue;
}
if (ov == __sync_val_compare_and_swap(&bits_, ov, ov | MASK)) {
break;
}
}
其中 __sync_val_compare_and_swap()
是 GCC 的内置原子 CAS。在进入循环之前检测到 bits_ & mask
为 true
的情况下,GCC(合理地)将其优化为无限循环,完全跳过 CAS 操作,所以我建议以下更改(有效):
while (true) {
uint8_t ov = bits_; // bits_ is some "uint8_t" non-local variable
if (ov & MASK) {
__sync_synchronize();
continue;
}
if (ov == __sync_val_compare_and_swap(&bits_, ov, ov | MASK)) {
break;
}
}
在我回答之后,OP 注意到将 bits_
更改为 volatile uint8_t
似乎也可以。我建议不要走那条路,因为 volatile
通常不应该用于同步,而且在这里使用栅栏似乎没有太大的缺点。
但是,我考虑得更多,在这种情况下,语义使得 ov & MASK
检查是否基于陈旧值并不重要,只要它是不是基于无限陈旧的(即只要循环最终被打破),因为更新 bits_
的实际尝试是同步的。如果 bits_
被另一个线程更新,例如 bits_ & MASK == false
,volatile
是否足以保证此循环最终终止,例如任何现有的处理器?换句话说,在没有显式内存栅栏的情况下,实际上是否可以无限期地由处理器有效地优化未由编译器优化的读取? (编辑:为了清楚起见,我在这里询问现代硬件实际上可能会做什么,假设读取是由编译器在循环中发出的,因此尽管表达它,但从技术上讲这不是语言问题在 C++ 语义方面是方便的。)
这是它的硬件角度,但要稍微更新它并使其也成为有关 C++11 内存模型的可回答问题,请考虑对上面代码的以下变体:
// bits_ is "std::atomic<unsigned char>"
unsigned char ov = bits_.load(std::memory_order_relaxed);
while (true) {
if (ov & MASK) {
ov = bits_.load(std::memory_order_relaxed);
continue;
}
// compare_exchange_weak also updates ov if the exchange fails
if (bits_.compare_exchange_weak(ov, ov | MASK, std::memory_order_acq_rel)) {
break;
}
}
cppreference声称 std::memory_order_relaxed
意味着“对围绕原子变量的内存访问的重新排序没有限制”,因此独立于实际硬件会或不会做什么,确实暗示 bits_.load( std::memory_order_relaxed)
可以在技术上永远在 bits_
在符合实现的另一个线程上更新后读取更新的值?
编辑:我在标准(29.4 p13)中找到了这个:
Implementations should make atomic stores visible to atomic loads within a reasonable amount of time.
显然,等待更新值“无限长”是不可能的(大多数情况下?),但没有硬性保证任何特定的新鲜时间间隔应该是“合理的”;不过,关于实际硬件行为的问题仍然存在。
最佳答案
C++11 原子处理三个问题:
确保在没有线程切换的情况下读取或写入完整的值;这样可以防止撕裂。
确保编译器不会在原子读取或写入的线程中重新排序指令;这确保了线程内的排序。
ensuring (for appropriate choices of memory order parameters) that data written within a thread prior to an atomic write will be seen by a thread that reads the atomic variable and sees the value that was written.这就是可见性。
当您使用 memory_order_relaxed
时,您无法从宽松的存储或加载中获得可见性保证。你确实得到了前两个保证。
实现“应该”(即鼓励)使内存写入在合理的时间内可见,即使是在宽松的顺序下也是如此。这是可以说的最好的了;这些东西迟早会出现的。
所以,是的,正式地,从不让轻松写入对轻松读取可见的实现符合语言定义。实际上,这不会发生。
关于 volatile
的作用,请咨询您的编译器供应商。这取决于实现。
关于c++ - volatile 但不 protected 读取能否产生无限期的陈旧值? (在真实硬件上),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15467042/
我正在寻找x86 Assembly中pow(real, real)的实现。我也想了解算法的工作原理。 最佳答案 只需将其计算为2^(y*log2(x))即可。 有一个x86指令FYL2X计算y *
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 11 年前。 Improve thi
风格指南的最后一点 http://www.python.org/dev/peps/pep-0008 读... 不要使用 == 将 bool 值与 True 或 False 进行比较。 为什么? 编辑只
我似乎无法完成这件事。我仍然在我的日志中看到 cloudflare IP。目前,我有一个负载均衡器,它位于 Cloudflare 后面。 目前,这是与 forwardfor 相关的块: opt
此代码行选择任何类名不是“id”和“quantity”的 div 内的所有子输入:: $("div.item > div:not(.id,.quantity) > :input").live("key
我在测试真实产品时遇到错误。当我单击购买按钮时,出现错误“此版本的应用程序未配置为市场计费”。 我用这个例子https://github.com/robotmedia/AndroidBillingLi
到目前为止我能找到的所有答案都建议调用 omp_set_num_threads。虽然在大多数情况下这是一个正确的答案,但它对我不起作用。在内部,调用 omp_set_num_threads 会导致创建
假设我有 3 个显示器。如何仅通过其索引获取第二个句柄? EnumDisplayMonitors() 不会工作,因为它也枚举了伪设备,而 EnumDisplayDevices() 没有给我句柄。 最佳
我一直在尝试制作一个简单的小游戏来测试我的逻辑,这是一个简单的迷宫,它很丑,而且到目前为止很糟糕。 引擎工作得很好,考虑到迷宫已经存在(矩阵),它甚至可以愉快,但我无意绘制一堆 map ,这可能是在
Cloudflare 代理包含一个名为 CF-Connecting-IP 的 header 和用户的真实 IP。我想让 traefik 读取这个 header 并用它的内容创建一个 X-Real-Ip
我想要上下文菜单中的不同菜单项,具体取决于我在 JTable 中单击的行 大多数示例并没有真正显示上下文菜单(应该根据上下文 - 所选行进行填充) 我尝试过这个: popupMenu = ne
我有一个对象 callInst。如何获取函数的真实名称而不是 IR 代码中的名称?如果我在我的通行证中运行此代码(Useless 在另一个问题中发布) StringRef get_function_n
我在 Appium 和 iPad 2 上使用了以下所需的功能 DesiredCapabilities capabilities = new DesiredCapabilities(); capabil
根据documentation 我们可以通过以下方式在模拟器上运行我们的 android 项目: cordova 运行 android 或 cordova emulate android 但是如何在真
在 ASP.NET 中,有没有办法获得真正的原始 URL? 例如,如果用户浏览到“http://example.com/mypage.aspx/%2F”,我希望能够获得“http://example.
我的 NSInputStream 遇到问题。这是我的代码: case NSStreamEventHasBytesAvailable: printf("BYTE AVAILABLE\n"
$(selector).click() 结果没有任何反应。 this answer在浏览器控制台中工作,javascript 上下文设置为 iframe,但不是主页: simulateMouseCli
我在我的 AB Micro820 PLC 中设置了 modbus 映射。我在 40001 中有一个数组用于写入,在 42001 中有一个数组用于读取。两者都是 200 个元素和 REAL 类型(32
我正在寻找有关设计契约(Contract)管理数据模型的建议。因此,合约的一般生命周期是: 契约(Contract)已创建并处于“草稿”状态。它可以在内部查看,并且可以进行更改。 契约(Contrac
我正在尝试让我的 WebView fullscreen 达到标准,我知道如何将它拉伸(stretch)到“全屏”,但我想做的是当您长按 WebView 并选择全屏选项时,创建全屏。有谁知道该怎么做?
我是一名优秀的程序员,十分优秀!