引用http://www.importnew.com/18126.html
volatile 修饰变量关键字,每次读取该变量时,强制从内存中读取,而不是从缓存或者寄存器中读取。确保本条指令不会被编译器所优化。
volatile 无法保证原子性操作,是非线程安全的。
一旦一个共享变量被volatile 修饰,那么具备两侧语义:
1、当该变量被修改后,其他线程立刻可见该变量的新值。
2、禁止进行指令重排序。
//线程1
boolean
stop =
false
;
while
(!stop){
doSomething();
}
//线程2
stop =
true
;
stop 被修饰成volatile 之后,stop 变量值发生变化,线程1 会立刻停止。这是对可见性的解释。(经过反复测试,stop并未定义为volatile,线程1也立刻停止,求大神解释)
1 2 3 4 5 6 7 8 | //x、y为非volatile变量 //flag为volatile变量 x = 2 ; //语句1 y = 0 ; //语句2 flag = true ; //语句3 x = 4 ; //语句4 y = - 1 ; //语句5 |
语句1和语句2 发生在语句3之前,语句4和语句5发生语句3之后,并且volatile关键字能保证,执行到语句3时,语句1和语句2必定是执行完毕了的,且语句1和语句2的执行结果对语句3、语句4、语句5是可见的
应用场景:
1、多线程需要操作共享变量,且共享变量的变化不依赖于当前值。
2、在单例模式中应用如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 | class Singleton{ private volatile static Singleton instance = null ; private Singleton() { } public static Singleton getInstance() { if (instance== null ) { synchronized (Singleton. class ) { if (instance== null ) instance = new Singleton(); } } return instance; } } |