volatile的用法
volatile首先只能用于类的成员变量,因为volatile的作用是保证多线程之间对变量的可见性。所以如果使用volatile关键字修饰局部变量,由于每一个线程都有自己的私有的栈空间,所以局部变量无法被其他的线程所访问。不volatile只能用于修饰类的成员变量。
代码案例(内存可见性)
java
package org.example.thread;
import java.util.Scanner;
public class volatileDemo {
public static int isQit = 2;
public static void main(String[] args) {
Thread t = new Thread(()->{
while(isQit!=0){
}
System.out.println("finished");
});
Thread t2 = new Thread(()->{
Scanner scanner = new Scanner(System.in);
System.out.println("please enter a number: ");
isQuit = scanner.nextInt();
});
t.start();
t2.start();
}
}
线程t死循环判断isQuit的值,但是while循环里什么也不做。线程t2用于修改isQuit的值,以达到终止线程t的循环的功能。但是执行后发现输入的值是0也无法终止线程t的循环。
但是在isQuit前面加上volatile关键字即可完成该功能。
内存可见性分析
为何加了volatile关键字就可以了呢??
因为线程t在判断时,编译器可能为了提高读取效率,本来isQuit变量应该被存储在内存中,等待编译器去读取,但是编译器为了提高读取效率,就会将isQuit的值直接读取到寄存器中。故每次进行循环时,读取的都是寄存器中的值,但是线程t2修改的isQuir变量的结果是直接写入到内存中,故线程t并没有感知到,所以一直进行死循环。
代码案例(指令重排序)
见文章:https://blog.csdn.net/weixin_73906153/article/details/156691700?spm=1001.2014.3001.5501 末尾
使用建议
只要是在同步代码块中需要被修改获取的成员变量,都尽量使用volatile关键字进行修饰,防止出现意料之外的问题。