变量的线程可见性。
在CPU计算过程中,会将计算过程需要的数据加载到CPU计算缓存中,当CPU计算中断时,有可能刷新缓存,重新读取内存中的数据。在线程运行的过程中,如果某变量被其他线程修改,可能造成数据不一致的情况,从而导致结果错误。而volatile修饰的变量是线程可见的,当JVM解释volatile修饰的变量时,会通知CPU,在计算过程中,每次使用变量参与计算时,都会检查内存中的数据是否发生变化,而不是一直使用CPU缓存中的数据,可以保证计算结果的正确。
volatile只是通知底层计算时,CPU检查内存数据,而不是让一个变量在多个线程中同步。
例:
/**
* volatile
*/
package com.sxt.concurrent.t02;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.TimeUnit;
public class Test_01 {
public static void main(String[] args) {
final Test_01_Container t = new Test_01_Container();
new Thread(new Runnable() {
@Override
public void run() {
for(int i = 0; i < 10; i++){
System.out.println("add Object to Container " + i);
t.add(new Object());
try {
TimeUnit.SECONDS.sleep(1);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
new Thread(new Runnable(){
@Override
public void run() {
while(true){
if(t.size() == 5){
System.out.println("size = 5");
break;
}
}
}
}).start();
}
}
class Test_01_Container{
//去掉 volatile 修饰符则另外一个线程无法实时获取该变量的变化
volatile List<Object> container = new ArrayList<>();
public void add(Object o){
this.container.add(o);
}
public int size(){
return this.container.size();
}
}
网友评论