- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
Java 通过其原子类公开 CAS 操作,例如
boolean compareAndSet(expected,update)
JavaDocs指定 compareAndSet 操作的内存效果如下:
compareAndSet and all other read-and-update operations such as getAndIncrement have the memory effects of both reading and writing volatile variables.
这绝对适用于成功的 compareAndSet
调用。但是,如果 compareAndSet 返回 false
,内存效应是否也成立?
我会说不成功的 compareAndSet
对应于 volatile 读取(因为在这种情况下必须访问原子实例的当前值),但我不明白为什么 CAS 应该执行不成功情况下的特殊内存屏障指令。
问题实际上是,不成功的 CAS 是否也建立了 happens-before 关系。考虑以下程序:
public class Atomics {
private static AtomicInteger ai = new AtomicInteger(5);
private static int x = 0;
public static void main(String[] args) {
new Thread(() -> {
while (x == 0) {
ai.compareAndSet(0, 0); // returns false
}
}, "T1").start();
new Thread(() -> {
x = 1;
ai.compareAndSet(0, 0); // returns false
}, "T2").start();
}
}
线程 T2(和程序)一定会终止吗?
最佳答案
使用 volatile
读取和写入建立happens-before 关系的问题在于,这种关系只存在于写入和后续 阅读。如果一个线程 T1 写入一个共享的 volatile
变量,而另一个线程 T2 从同一个变量读取,如果 T2 之前读取该变量,则不存在happens-before 关系T1 写信给它。如果决定 T1 是否在 T2 读取之前写入的只是线程调度,那么我们没有任何保证。
在没有额外同步的情况下处理它的一种实用方法是评估实际值,T2 已读取。如果此值表明 T1 已写入新值,则我们有一个有效的happens-before 关系。这就是它在使用 volatile boolean fooIsInitialized
标志或 volatile int currentPhase
计数器时的工作方式。很明显,如果写入的值与旧值相同,或者如果新值从未实际写入,这将无法工作。
您的示例程序的问题在于它推测线程调度。它假设 T2 最终执行了 cas 操作,并且在 T1 中将有一个后续迭代,其中下一个 cas 将创建一个先发生关系。但这并不能保证。这可能不是直觉上可以理解的,但如果没有同步,T1 的所有迭代都可能在 T2 的操作之前发生,即使循环是无限的。它甚至是一种有效的线程调度行为,让 T1 永远消耗 100% 的 CPU 时间,然后再将 CPU 时间分配给 T2,因为无法保证在相同优先级的线程之间进行抢占式线程切换。
但是,即使底层系统确实将 CPU 时间分配给最终会执行操作的 T2,JVM 也不需要让 T1 明白这一点,因为 T1 无法观察到 T2 曾经运行过。在现实生活中不太可能发现这一点,但答案仍然是没有保证。当存在一系列操作使 T1 可以观察到 T2 运行(即更改其状态)时,情况会发生变化,但当然,该操作链会使 cas 过时。
关于java - compareAndSet 不成功操作的内存效应,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29846319/
我正在使用 jQuery 的 $.ajax 函数来提交表单,它可以工作,但成功正是我遇到问题的地方。这是我的代码: $("#form").submit(function () { $.
我正在使用动态分页。 我需要在开始另一个事件之前取消 jQuery ajax 中的 success 事件。 我已经设置了一个等于$.ajax()的变量,在这样做之前,无论如何我都会调用abort。 问
如果我错了,请纠正我,但我对 $.post 成功/失败的理解是,如果 url 有效,这将返回成功。唯一会返回失败的情况是 url 无效。 如果这是真的,我如何验证成功函数?我问的原因是无论发生什么,即
HANDLE hFile = CreateFile(LPCTSTR("filename"), // name of the write
我正在使用以下代码发送短信。但这似乎不会在未发送短信时产生异常。例如,当没有足够的钱发送时,我仍然会去 smsSucces();有没有人知道解决此问题的方法以确保它已发送? private b
我正在尝试将字符串转换为 DateTime,在一台计算机上,它工作正常,但在另一台计算机上,它却不行!它运行的计算机运行的是 32 位 Windows 7,它不运行的计算机运行的是 64 位 Wind
我在页面上使用表单让用户输入将用于各种目的的图像的 url。我正在编写一个 ajax 方法来确定他们提供的 url 是否实际上是图像。到目前为止,我已经这样做了: $(document).on('re
我在 jquery 中对 php 脚本进行 ajax 调用。但是 php 脚本需要返回什么才能触发 ajax 中的成功/错误处理程序。所以这是 ajax: $.ajax({ data:
几个简单的问题: 对于 native 和 Flash/Silverlight 垫片来说,成功事件是“规范化”事件吗?记录的示例表明它仅适用于 Flash/Silverlight 对象准备就绪的情况。
这个问题不太可能对任何 future 的访客有帮助;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况相关,通常不适用于互联网的全局受众。如需帮助使这个问题更广泛适用,visit the h
我尝试使用新的 Groovy Grape Groovy 1.6-beta-2 中的功能,但我收到一条错误消息; unable to resolve class com.jidesoft.swing.J
我正在使用 sequelize/nodejs/express/react 将实体持久化到 postgres 数据库 我有两个主要模型,国家和事件,我正在使用该应用程序,并且有一个名为“保存到数据库”的
我有以下代码,其中有 2 个电子邮件输入字段,我需要验证它们是否相同,并且使用 jQuery validate equalTo 成功运行。 Email Address
我正在尝试找出解决此问题的正确方法。 假设我们有一家元素商店。这些项目可以编辑、删除和创建。编辑或添加项目时,路线更改为/item/add 或/item/edit/{id}。 在 saga 成功添加或
这个问题已经有答案了: How do I return the response from an asynchronous call? (42 个回答) 已关闭 8 年前。 我有这段代码,警报工作正常
Closed. This question needs to be more focused。它当前不接受答案。 想改善这个问题吗?更新问题,使其仅关注editing this post的一个问题。
我想在单击超链接 (.remove_resort) 时(成功的 ajax 调用后)删除超链接的(父)跨度。 虽然ajax调用成功,但是最后span并没有被移除。这里出了什么问题? 请记住:有几个类
我正在编写一个非常简单的程序来将鼠标剪辑到指定的窗口。它从系统托盘运行,没有可见窗口。由于同一窗口会有多个实例,因此它使用 EnumWindows() 迭代每个顶级窗口,并将它们的 hwnd 与 Ge
我正在尝试找出如何执行 if 语句,以便如果玩家的击球率超过 0.250,则会为成功的 tr 添加一个类别。 我发现了以下堆栈问题,但我不确定可以使用或应该使用哪种方式以及如何使用这些堆栈问题。 ht
我是 Prolog 的新手,我正在尝试解决这个练习: Define a predicate greater_than/2 that takes two numerals in the notation
我是一名优秀的程序员,十分优秀!