volatile 用处说明
在 JDK1.2 之前,Java 的内存模型实现总是从主存(即共享内存)读取变量,是不需要进行特别的注意的。而随
着 JVM 的成熟和优化,现在在多线程环境下 volatile 关键字的使用变得非常重要。
在当前的 Java 内存模型下,线程可以把变量保存在本地内存(比如机器的寄存器)中,而不是直接在主存中进
行读写。这就可能造成一个线程在主存中修改了一个变量的值,而另外一个线程还继续使用它在寄存器中的变量值
的拷贝,造成数据的不一致。
要解决这个问题,就需要把变量声明为 volatile,这就指示 JVM,这个变量是不稳定的,每次使用它都到主存中
进行读取。一般说来,多任务环境下,各任务间共享的变量都应该加 volatile 修饰符。
Volatile 修饰的成员变量在每次被线程访问时,都强迫从共享内存中重读该成员变量的值。而且,当成员变量发
生变化时,强迫线程将变化值回写到共享内存。这样在任何时刻,两个不同的线程总是看到某个成员变量的同一值。
Java 语言规范中指出:为了获得最佳速度,允许线程保存共享成员变量的私有拷贝,而且只当线程进入或者离
开
同步代码块时才将私有拷贝与共享内存中的原始值进行比较。
这样当多个线程同时与某个对象交互时,就必须注意到要让线程及时的得到共享成员变量的变化。而 volatile 关键字就是提示 JVM:对于这个成员变量,不能保存它的私有拷贝,而应直接与共享成员变量交互。
volatile 是一种稍弱的同步机制,在访问 volatile 变量时不会执行加锁操作,也就不会执行线程阻塞,因
此 volatilei 变量是一种比 synchronized 关键字更轻量级的同步机制。
使用建议:在两个或者更多的线程需要访问的成员变量上使用 volatile。当要访问的变量已在
synchronized 代码块中,或者为常量时,没必要使用 volatile。
由于使用 volatile 屏蔽掉了 JVM 中必要的代码优化,所以在效率上比较低,因此一定在必要时才使用此关
键字。