(Java 多线程系列)java volatile详解

时间:2023-06-20 00:00:56

  在前面的文章里面介绍了synchronized关键字的用法,这篇主要介绍volatile关键字的用法。

  Java语言提供了一种稍弱的同步机制,即volatile变量,用来确保将变量的更新操作通知到其它线程。当把变量声明为volatile类型后,编译器与运行时都会注意到这个变量是共享的,因此不会将该变量上的操作与其它内存操作一起重排序。Volatile变量不会被缓存在寄存器或者其它对处理器不可见的地方,因此在读取volatile类型的变量时,总会返回最新写入的值。

volatile boolean asleep;
while(!asleep){
//do something
}

volatile变量通常用做某个操作完成、发生中断或者状态的标志。虽然volatile变量使用很方便,但是存在一些局限性:volatile的语义不足以确保递增操作(count++)的原子性,除非你能确保只有一个线程对变量执行写操作。

  volatile与加锁机制的主要区别是:加锁机制既可以确保可见性又可以确保原子性,而volatile变量只有确保可见性。

下面以一个实例来说明使用volatile变量作为状态标志在多线程里面的使用。

public class VolatileTest{
volatile boolean mFlag = false;
public static void main(String[] args) {
final VolatileTest mVolTest = new VolatileTest();
Thread t1 = new Thread(new Runnable() {
public void run() {
while(!mVolTest .mFlag ){
System.out.println("Working ...");
try {
Thread.sleep(1000);
} catch (InterruptedException ie) {
//
}
}
}
}, "t1");
Thread t2 = new Thread( new Runnable() {
public void run() {
try {
Thread.sleep(3000); //3秒后停止线程t1
mVolTest.mFlag = true;
} catch (InterruptedException ie) {
//
}
}
}, "t2");
t1.start();
t2.start();
}
}

  当且仅当满足以下所有条件时,才应该使用volatile变量:

  a)对变量的写入操作不依赖变量的当前值,或者你能确保只有单个线程更新变量的值

  b)该变量不会与其它状态变量一起纳入不变性条件中

  c)在访问变量时不需要加锁