JAVA中的volatile以及transient的差异
JAVA中的volatile以及transient的区别
在JVM1.2之前,JAVA的内存模型总是从主存读取变量,所以不需要特别注意。
在当前的JAVA内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进行读写,这就可能造成了一个线程在主存中修改了变量的值,而另外一个线程却还是使用了寄存器中的拷贝,造成了数据的不一致。
解决这个问题的方法是:只需要把变量声明为volatile类型即可。这就指示JVM,这个变量是不稳定的,每次使用它都要到主存中进行读取。
Volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值,并且当成员变量发送变化时,强迫线程将变化值回写到共享内存里。这样在任何时刻,2个不同的线程总是看到某个成员变量的同一个值。
JAVA语言规范指出:为了获取最佳读写速度,运行的多个线程可以保存共享的成员变量的私有拷贝,只有当线程进入或离开同步块的时候才与成员变量的原始值进行对比。
这样当多个线程同时与某个对象交互的时候,就必须要注意到要让线程及时得到共享成员变量的变化。
而volatile关键字就是提示VM:对于这个成员变量,不能保存它的私有拷贝,而应直接与共享变量交互。
使用建议:在2个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。由于使用volatile屏蔽了VM中必要的代码优化,所以在效率上比较低。
在JVM1.2之前,JAVA的内存模型总是从主存读取变量,所以不需要特别注意。
在当前的JAVA内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进行读写,这就可能造成了一个线程在主存中修改了变量的值,而另外一个线程却还是使用了寄存器中的拷贝,造成了数据的不一致。
解决这个问题的方法是:只需要把变量声明为volatile类型即可。这就指示JVM,这个变量是不稳定的,每次使用它都要到主存中进行读取。
Volatile修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值,并且当成员变量发送变化时,强迫线程将变化值回写到共享内存里。这样在任何时刻,2个不同的线程总是看到某个成员变量的同一个值。
JAVA语言规范指出:为了获取最佳读写速度,运行的多个线程可以保存共享的成员变量的私有拷贝,只有当线程进入或离开同步块的时候才与成员变量的原始值进行对比。
这样当多个线程同时与某个对象交互的时候,就必须要注意到要让线程及时得到共享成员变量的变化。
而volatile关键字就是提示VM:对于这个成员变量,不能保存它的私有拷贝,而应直接与共享变量交互。
使用建议:在2个或者更多的线程访问的成员变量上使用volatile。当要访问的变量已在synchronized代码块中,或者为常量时,不必使用。由于使用volatile屏蔽了VM中必要的代码优化,所以在效率上比较低。