- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
Hazard pointers是一种在无垃圾收集的情况下在无锁代码中安全回收内存的技术。
这个想法是,在访问一个可以并发删除的对象之前,一个线程设置它的冒险指针指向那个对象。想要删除对象的线程将首先检查是否设置了任何冒险指针指向该对象。如果是这样,删除将被推迟,这样访问线程就不会以读取已删除的数据而告终。
现在,假设我们的删除线程开始迭代危险指针列表,并在 i+1
元素处被抢占。现在,另一个线程将 i
处的风险指针设置为删除线程当前试图删除的对象。之后,删除线程恢复,检查列表的其余部分,并删除该对象,即使现在在位置 i
处有一个指向该对象的冒险指针。
很明显,仅仅设置危险指针是不够的,因为删除线程可能已经检查了我们的危险指针并决定我们的线程不想访问该对象。我如何确保在设置危险指针后,我尝试访问的对象不会从我手中删除?
最佳答案
original paper by Maged M. Michael对使用危险指针的算法施加了这一重要限制:
The methodology requires lock-free algorithms to guarantee that no thread can access a dynamic node at a time when it is possibly removed from the object, unless at least one of the thread’s associated hazard pointers has been pointing to that node continuously, from a time when the node was guaranteed to be reachable from the object’s roots. The methodology prevents the freeing of any retired node continuously pointed to by one or more hazard pointers of one or more threads from a point prior to its removal.
正如 Anton's answer 中指出的那样, 删除是一个两阶段操作:首先你必须“取消发布”节点,将它从数据结构中删除,这样它就不能再从公共(public)接口(interface)访问。
此时,按照 Michael 的说法,节点可能被移除。并发线程访问它不再安全(除非它们一直持有指向它的危险指针)。
因此,一旦一个节点可能被删除,删除线程迭代风险指针列表是安全的。即使删除线程被抢占,并发线程也可能不再访问该节点。在验证没有风险指针被设置到节点之后,删除线程可以安全地进行到删除的第二阶段:实际的释放。
综上所述,删除线程的操作顺序是
D-1. Remove the node from the data structure.
D-2. Iterate the list of hazard pointers.
D-3. If no hazards were found, delete the node.
真正的算法稍微复杂一些,因为我们需要维护一个无法回收的节点列表,并确保它们最终被删除。这已被跳过,因为它与解释问题中提出的问题无关。
设置危险指针不足以保证安全访问它。毕竟,当我们设置风险指针时,该节点可能已被删除。
确保安全访问的唯一方法是,如果我们能够保证我们的风险指针一直指向该节点,从保证可以从对象的根访问该节点。
由于代码应该是无锁的,只有一种方法可以实现这一点:我们乐观地将风险指针设置为节点,然后检查该节点是否已被标记为可能已删除(即,它是不再可以从公共(public)根访问)之后。
因此访问线程的操作顺序是
A-1. Obtain a pointer to the node by traversing the data structure.
A-2. Set the hazard pointer to point to the node.
A-3. Check that the node is still part of the data structure.
That is, it has not been possibly removed in the meantime.
A-4. If the node is still valid, access it.
在一个节点可能被删除后(D-1
),删除线程可能被抢占。因此,并发线程仍然可以乐观地将它们的危险指针设置为它(即使不允许它们访问它)(A-2
)。
因此,删除线程可能会检测到虚假危险,从而阻止它立即删除该节点,即使其他线程都不会再访问该节点。这将以与合法危险相同的方式简单地延迟节点的删除。
重点是节点最终还是会被删除
在验证节点未被潜在删除之前,访问线程可能会被删除线程抢占 (A-3
)。在这种情况下,不再允许访问该对象。
请注意,如果抢占发生在 A-2
之后,访问线程访问该节点甚至是安全的(因为自始至终都有一个指向该节点的风险指针),但是由于访问线程无法区分这种情况,因此它一定会虚假地失败。
重要的一点是一个节点只有在它没有被删除的情况下才会被访问。
关于multithreading - 使用危险指针的无锁内存回收,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25204494/
我有2个功能: function func1() while true do -- listen on connection end end function func2()
我的问题可能看起来很奇怪,但我想我正面临着 volatile 的问题。对象。 我写了一个这样实现的库(只是一个方案,不是真正的内容): (def var1 (volatile! nil)) (def
由于 maven 支持多线程构建,是否可以同时运行 Sonar 多线程? (例如 mvn sonar:sonar -T 4 ) 我运行了它,当模块报告成功时,它报告整个构建失败并返回 java.uti
我们正在启动一个网站,该网站在短时间内的交易量非常大。它基本上是在给票。该代码是用Java,Spring和Hibernate编写的。我想通过产生多个线程并尝试使用JUnit测试用例来获取票证来模仿高容
我正在尝试访问像素数据并将图像从游戏中的相机保存到磁盘。最初,简单的方法是使用渲染目标,然后使用RenderTarget-> ReadPixels(),但是由于ReadPixels()的 native
我们有以下系统: 用户数:〜500k 项目数:〜100k UserSimilarity userSimilarity = new TanimotoCoefficientSimilarity(dataM
也许这是一个经常出现的问题,但我需要根据我的上下文进行一些自定义。 我正在使用 Spring Batch 3.0.1.RELEASE 我有一个简单的工作,有一些步骤。一个步骤是这样的 block :
也许这是一个经常出现的问题,但我需要根据我的上下文进行一些自定义。 我正在使用 Spring Batch 3.0.1.RELEASE 我有一个简单的工作,有一些步骤。一个步骤是这样的 block :
我正在尝试使用PyBrain和Python的multiprocessing软件包在Python中训练神经网络。 这是我的代码(它训练了一个简单的神经网络来学习XOR逻辑)。 import pybrai
我有一个繁重的功能,不适合在主时间轴上执行(因为要花很长时间才能完成并使程序崩溃)。 因此我在air(as3)中搜索多线程,但是我发现的所有示例都说明了如何在worker中运行单独的swf文件。如何在
我想实现线程A 和线程B 并行运行并共享全局变量。 下面是用python编写的代码。我想在中执行相同操作Dart (我不想使用future等待,因为它正在等待其他线程完成或必须等待。) 大小写变量:
我的一个项目只适用于调试 DLL,而不适用于非调试 DLL。 在 Debug DLL 设置下发布项目有哪些注意事项?例如,是否丢失了某些优化? 如何通过将调试版本设置为非调试 DLL 来调试此项目?我
我正在尝试比较 Matlab 和 Julia 之间的速度和性能。我正在查看一个代码,该代码对承受给定负载的连续体结构进行拓扑优化。我正在查看的代码是公共(public)代码topopt88.m:htt
Serving Flask 应用程序“服务器”(延迟加载) 环境:生产警告:这是一个开发服务器。不要在生产部署中使用它。请改用生产 WSGI 服务器。 Debug模式:开启 在 http://0.0.
我对 PyQT 很陌生。我正在学习如何制作 Progressbar 并随着算法的进展对其进行更新。我已经能够制作一个使用此链接进行 self 更新的基本进度条:Python pyqt pulsing
我正在尝试指定在特定线程上运行任务,这样我就可以使用两个专用于“放入” channel 的耗时任务的线程,而其他线程则用于处理该任务。 我对如何将特定任务分配给特定线程感到困惑。我以为我可以使用类似
我正在编写一个软件,它对很多(潜在的大)图像进行大量图像操作/合成。 多线程有助于提高速度,但 QT 不允许同时在同一图像上使用多个 QPainter。 所以我必须在副本的每个线程中进行图像操作/合成
此脚本读取 url 文件以执行多线程 HTTP 请求。 如何使用带有 url 的数组来发出多线程请求? 我的阵列将有类似的东西: @array = ("https://example.com/xsd"
Java 文档声明了以下关于构造函数同步的内容: Note that constructors cannot be synchronized — using the synchronized keyw
我有一个程序,其中主线程创建了很多线程。它崩溃了,我正在调试核心文件。崩溃发生在其中一个子线程中。为了找到原因,我需要知道主线程是否还活着。有什么方法可以找出哪个线程是初始线程? 最佳答案 Is th
我是一名优秀的程序员,十分优秀!