java用volatile修饰变量来确保变量的更新对其他线程是可见的,当变量用volatile关键字修饰后,编译器与运行时都会注意到这个变量是共享的,因此不会将该变量上的操作与其他内存操作一起重排序,即不再做优化,volatile变量不再存储到寄存器或对其他处理器不可见的地方,不管哪一个线程修改了volatile变量,都不会缓存而是直接更新到内存里面去,当线程读取volatile变量时,也是直接从内存中读取最新值,因此volatile变量的更新对所有 线程都是可见的。 volatile变量的一种典型的用法:检查某个状态标记以判断是否退出循环。如下如果asleep不是volatile变量,当asleep被另一个线程修改状态后,执行循环判断的线程却发现不了。
volatile boolean asleep;
...
while(!asleep)
countSomeSheep();
当线程A首先写入一个volatile变量并且线程B随后读取该变量时,在写入volatile变量之前对A可见的所有变量的值,在B读取了volatile变量后,对B也是可见的。如下面程序中,num变量在change变量之前改变的,所以对其他线程也是立即可见的。
public class VolatileTest implements Runnable{
private int num;
private volatile boolean change;
public VolatileTest(){
this.num = 10;
this.change = false;
}
public void run(){
while(num == 10){
int currNum = num;
if(change){
Object o = new Object();
synchronized(o){
}
int aftSycNum = num;
System.out.println("currNum = " + currNum + "," + "change = " + change);
System.out.println("afterSycNum = " + aftSycNum + "," + "change = " + change);
}
}
}
public static void main(String[] args) throws InterruptedException {
VolatileTest vt = new VolatileTest();
Thread t = new Thread(vt);
t.start();
Thread.sleep(100);
vt.num = 11;
Thread.sleep(100);
vt.change = true;
}
}
volatile关键字只是保证了变量的可见性,但并不保证变量的多线程环境下的安全性,因此使用的时候一般满足下面的条件:
1.对变量的修改操作不依赖于变量的当前值,或是确保只有单个线程更新变量的值。
2.在访问变量是不需要加锁。
3.该变量不会与其他变量一起纳入不变性条件中。