volatile关键字在c语言中也有。
volatile是一个类型
修饰符
(type specifier),就像大家更熟悉的const一样,它是被设计用来修饰被不同线程访问和修改的
变量
。
volatile
的作用是作为指令
关键字
,确保本条指令不会因
编译器
的优化而省略,且要求每次直接读值。
简单地说就是防止编译器对代码进行优化。比如如下程序:
XBYTE[2]=0x55;
XBYTE[2]=0x56;
XBYTE[2]=0x57;
XBYTE[2]=0x58;
对外部硬件而言,上述四条语句分别表示不同的操作,会产生四种不同的动作,但是编译器却会对上述四条语句进行优化,认为只有XBYTE[2]=0x58(即忽略前三条语句,只产生一条机器代码)。如果键入
volatile
,则编译器会逐一地进行编译并产生相应的机器代码(产生四条代码)。
精确地说就是,优化器在用到这个变量时必须每次都小心地重新读取这个变量的值,而不是使用保存在
寄存器
里的备份。
java中的volatile的作用好像是和C语言中的作用是一样的。
比如下面的程序,将不会结束,我的预期是会结束的。
public class RunThread extends Thread{
private boolean isRunning = true;
private void setRunning(boolean isRunning){
this.isRunning = isRunning;
}
public void run(){
System.out.println("进入run方法..");
int i = 0;
while(isRunning == true){
//..
}
System.out.println("线程停止");
}
public static void main(String[] args) throws InterruptedException {
RunThread rt = new RunThread();
rt.start();
Thread.sleep(1000);
rt.setRunning(false);
System.out.println("isRunning的值已经被设置了false");
}
}
进入run方法..
isRunning的值已经被设置了false
要想isRunning程序停掉,需要将 isRunning变量修饰为volatile变量即可。
public class RunThread extends Thread{
private volatile boolean isRunning = true;
private void setRunning(boolean isRunning){
this.isRunning = isRunning;
}
public void run(){
System.out.println("进入run方法..");
int i = 0;
while(isRunning == true){
//..
}
System.out.println("线程停止");
}
public static void main(String[] args) throws InterruptedException {
RunThread rt = new RunThread();
rt.start();
Thread.sleep(1000);
rt.setRunning(false);
System.out.println("isRunning的值已经被设置了false");
}
}
进入run方法..
isRunning的值已经被设置了false
线程停止
重要结论:volatile只能保证变量在线程之间的可见性,不能保证其原子性。也就是说,复值操作是线程安全的。多于复值就不是线程安全的。