- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
在 proposed boost::concurrent_unordered_map 的线程清理过程中出现了一些非常奇怪的情况并且是 recounted at this blog post .简而言之,bucket_type 是这样的:
struct bucket_type_impl
{
spinlock<unsigned char> lock; // = 2 if you need to reload the bucket list
atomic<unsigned> count; // count is used items in there
std::vector<item_type, item_type_allocator> items;
bucket_type_impl() : count(0), items(0) { }
...
然而线程清理器声称在 bucket_type 的构造和它的第一次使用之间存在竞争,特别是当 count atomic 被加载时。事实证明,如果你通过它的构造函数初始化一个 std::atomic<>,that initialisation is not atomic因此内存位置不是原子释放的,因此对其他线程不可见,鉴于它是原子的,这是违反直觉的,并且大多数原子操作默认为 memory_order_seq_cst。因此,您必须在构造后显式执行释放存储,以使用其他线程可见的值初始化原子。
是否有一些非常紧迫的原因为什么带有值消耗构造函数的 std::atomic 不使用释放语义初始化自身?如果不是,我认为这是一个库缺陷。
编辑: Jonathan 的回答更适合历史原因,但 ecatmur 的回答链接到 Alastair 关于此事的缺陷报告,以及它是如何通过简单地添加注释来说明施工报价而关闭的其他线程看不到。因此,我将把答案授予 ecatmur。感谢所有回答的人,我认为要求额外构造函数的方式很清楚,至少在文档中会突出显示值消耗构造函数存在一些不寻常的地方。
编辑 2:我最终向委员会提出这是 C++ 语言的一个缺陷,并发部分的主席 Hans Boehm 认为这不是问题,原因如下:
2014 年目前没有 C++ 编译器将消费视为与获取不同。因为在现实世界的代码中,你永远不会在不经过一些释放/获取的情况下将原子传递给另一个线程,所以原子的初始化将对使用原子的所有线程可见。我认为在编译器 catch 之前这很好,在此之前线程清理器会对此发出警告。
如果您像我一样执行不匹配的消费获取释放(我正在使用 release-inside-lock/consume-outside-lock 原子来推测性地避免不必要的释放获取自旋锁)那么你是一个足够大的男孩,知道你必须在构建后手动存储释放原子。这可能是一个公平的观点。
最佳答案
因为转换构造函数是constexpr
,constexpr
函数不能有原子语义等副作用。
在 DR846 , Alastair Meredith 写道:
I'm not sure if the initialization is implied by use of constexpr keyword (which restricts the form of a constructor) but even if that is the case, I think it is worth spelling out explicitly as the inference would be far too subtle in that case.
该缺陷的解决方案(由 Lawrence Crowl 提出)是用注释记录构造函数:
[Note: Construction is not atomic. —end note]
该注释随后扩展为当前的措辞,在 DR1478 中给出了一个可能的内存竞争的示例(通过 memory_order_relaxed
操作传达原子的地址) .
转换构造函数需要是 constexpr
的原因是(主要)允许静态初始化。在 DR768我们看到:
Further discussion: why is the ctor labeled "constexpr"? Lawrence [Crowl] said this permits the object to be statically initialized, and that's important because otherwise there would be a race condition on initialization.
所以:使构造函数 constexpr
消除了静态生命周期对象上的竞争条件,代价是动态生命周期对象中的竞争只发生在相当人为的情况下,因为要发生竞争,动态生命周期原子对象的内存位置必须以不会导致原子对象的 value 也同步到该线程的方式传递给另一个线程.
关于c++ - 为什么 std::atomic 初始化不进行原子释放,以便其他线程可以看到初始化值?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25609858/
如何获取特定 UTF-8 字符串的字节序列?我看到正则表达式引擎中的某些错误仅在某些边缘情况下才会触发,我想确切地知道它正在处理哪些数据。 最佳答案 也许 charToRaw ? > charToRa
考虑一个具有许多变量的 java 程序,其中一些是循环计数器,我希望看到这些变量随时间变化的值,而不是在代码中到处放置 print 语句。 为什么?我认为它可以帮助我轻松调试。 Example- in
有没有办法查看 JavaScript 正在创建的“线程”?例如,如果我有一个附加到 DOM 元素的事件处理程序,我假设 JavaScript 会隐式地创建一个新线程来在后台运行该代码?如果是这样,有没
我正在使用反射 API 来调用类运行时并调用其方法。但我看到 java.lang.reflect.InitationTargetException 异常。 我需要调用一个名为 - TestClass
考虑一个名为 t.cmd 的命令脚本,它只包含以下两行: @exit /b 123 @echo If you see this, THEN EXIT FAILED.. 因此,该脚本只是将脚本执行过程的
我最近一直在努力学习编程语言实用学第 3 版,以了解更多关于语言在底层如何工作的信息,并且通过查看由真正基本的 GCC 编译的 C 代码生成的程序集,我获得了很多进展.我开始对 C 系列的静态语言越来
最简单的制作方法是什么QWebView识别并正确加载网页上的 Flash? 最佳答案 似乎只需两行即可实现(当然,假设目标系统上安装了闪存): QWebSettings *settings = QWe
我想通过编程(使用VBA)访问其他人与我共享的日历。它们在我的Outlook中的“人民日历”下列出。我在网上搜索了此内容,所有建议所做的仅使我感到困惑。如何从“人的日历”中获取共享给我的所有日历的
使用MongoDB Compass,可以看到数据库用户或创建新用户吗? 最佳答案 香港专业教育学院在命令行上使用以下命令进行管理: mongo use (my database name) db.cr
我正在CentOS 7.4.1708(核心),Java(TM)SE运行时环境(内部版本1.8.0_152-b16)上运行eXist-db 3.6.1。我希望能够在person.xml中的最后一个记录之
我需要确认。 客户端 1 在事务内的表中插入行。 客户端 2 使用 SELECT 请求该表。如果在此客户端上隔离级别设置为 READ COMMITTED,您能否确认 SELECT 不会返回客户端 1
我刚刚安装了 python-dev: $ sudo apt-get install python-dev 这样我就可以开始使用 P4Python 了。但是当我尝试导入 P4 时,我得到: Traceb
我正在使用 msys 和 mingw 编译 libtorrent-rasterbar-0.16.16。./configure 运行良好,直到它进入 boost 库检查。我有 boost 1.51,我设
我在 GO 项目的 Travis CI 上有一个奇怪的行为。 [这里] 失败了,提示一个函数只接受 1 个参数并用 2 个参数调用。 src/finances-service/main.go:45:1
这个问题已经有答案了: What are enums and why are they useful? (27 个回答) 已关闭 4 年前。 作为 Java 初学者,我接触到了枚举,这让我对类型声明感
关闭。这个问题是off-topic .它目前不接受答案。 想改进这个问题吗? Update the question所以它是on-topic用于堆栈溢出。 关闭 11 年前。 Improve thi
我正在使用 preactjs 创建我的应用程序。最重要的是,我正在使用剑道网格。在网格内,我想显示一个超链接。如果用户点击链接,它应该改变路线。为了呈现链接,我使用了 preact-router。 这
有没有办法只查看 Node.js 中的函数签名?我知道 fn.toString() 但不需要所有代码,只需要签名。 最佳答案 var source = fn.toString(); var signa
如何设置 vim 以查看我在 ubuntu 上安装的 ruby rvm。 rvm 已安装,ruby 从终端返回。 renshaw@renshaw-TravelMate-5740G:~$ ruby
我试图在我的 Android 应用程序中防止 OutOfMemoryError。我已经阅读了很多帖子,但我仍然无法解决。 该应用程序有后台 Activity ,所以我认为这是主要问题。 OutOfMe
我是一名优秀的程序员,十分优秀!