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指定的,虽然我想链接到那个,但另一个答案已经做了 – 去推荐它。