java中的同步重新排序
众所周知,JVM不应该将带有同步块的语句重新排序到同步块之外。 考虑到这一点,是否允许JVM重新排序赋值y = 7
以便在以下代码段中的synchronized
块之后发生?
x = 5; y = 7; synchronized (this) { x = 6; }
我们知道在同步块之前的变量赋值可以重新排序以在块内发生。 所以以下内容应该是初始代码的有效重新排序:
x = 5; synchronized (this) { x = 6; y = 7; }
有人可能会争辩说,因为这是一个有效的排序,所以在synchronized
块之后不会发生y
赋值,因为它违反了同步块内的代码不能在块之后重新排序并推断y
发生的规则 – 在结束之前同步块的。
另一方面,可能所有的排序都不等同,重要的是哪个排序是实际排序。 具体来说,如果y
赋值最初是在同步块内完成的,则在块之后不会发生,否则就可能发生。
总结一下,接下来是订购第一个片段的有效排序吗?
x = 5; synchronized (this) { x = 6; } y = 7;
JLS 17.4.5 :
- 如果x和y是同一个线程的动作,并且x在程序顺序中出现在y之前,那么hb(x,y)。
…
- 如果动作x与后续动作y同步,那么我们也有hb(x,y)。
只有在假设y
的值在当前线程之外可见时,您的问题才有意义。 如果是这种情况,则这两个规则的组合要求在同步块之后不重新分配赋值。
是的,你的推理是有缺陷的; 这不可能发生。
监视器输入就像一个volatile load
(不完全正确,但我理解它更简单 – 将插入两个障碍: LoadLoad|LoadStore
)和之前的操作无法跨越该障碍。
我很确定,这是由JLS
指定的,虽然我想链接到那个,但另一个答案已经做了 – 去推荐它。