麻烦帮忙看一下,为什么会有n!=n这种现象?
java虚拟机性能的提高,有一部分得归功于对代码的重排序。在if(n!=n)中,其实存在控制依赖关系。这里有几个操作:
一、读取左边的n值,
二、读取右边的n值,
三、进行不等比较,
四、把不等比较的结果给if。
上面四个操作,由于依赖关系,有一二先于三,三先于四这样的执行顺序。而一二之间没有依赖关系,所以一二可以重排序。
所以可能会这样:
temp1=n,
temp2=n,
flag=temp1!=temp2
if(flag)
而执行上面四个操作的时候,可能另一个线程正在执行new操作,由于重排序,这个线程可能在对象未完全构造完毕就把对象的引用给发布出去了。
于是temp1=n操作可能发生在对象构造完毕前,此时可能构造器里面的this.n=n还未执行,所以此时的n可能是0,而不是传给构造函数的那个参数的值,而temp2=n则可能发生在对象构造完毕之后,此时的n是传给构造函数参数的值,于是最后就看到不一致的结果,导致n!=n为true
private int n;
n的初始值为0
当调用构造函数时,this.n=n;
n的值又会改变
n!=n就发生在改变前和改变后
很深奥啊.
我猜想可能这个类的示例是托管孵化出来的,
就是类似于spring这种管理,
构造参数是外部注入的.
所以这个时候有可能出现这种问题
但是这种问题出现的几率应该很低很低.
因为初始化构造函数的数值,跟调用非同步,请查看本篇http://blog.csdn.net/reboot123/article/details/51507330 由于n变量是非同步的,所有它的
assign和load过程可能是非同步的,如果给它加上volatile标识,就不会有这种情况出现了。