关键字volatile的主要作用是使变量在多个线程可见。
volatile保证了可见性
我们先来看一下java线程的工作内存与主内存的关系
从图中我们会发现,当多个线程操作同一个变量,会引起数据不一致的情况,比如线程A修改了一个变量,但还没有来得及写回主内存中,而线程B这时候从主内存中获取的变量则不是最新的值,还是原来的值。而利用volatile关键字修饰之后,则不会出现这个问题。java内存模型是通过在变量修改后将新值同步回主内存,在变量读取前从主内存刷新变量。前文说的volatile关键字的可见性,就是修改后的新值能立即同步回主内存,以及每次使用 前立即从主内存刷新,因此可以说volatile保证了多线程操作时变量的可见性,而普通变量则不能保证这一点。
volatile不保证原子性
volatile修饰的变量,并不具备原子性,我们举个例子来说明一下
public class VolatileTest extends Thread{
volatile private static int count;
public void run(){
for(int i=0; i<100; i++){
count++;
}
System.out.println("count="+count);
}
public static void main(String[] args) {
Thread[] threadArray = new Thread[100];
for(int i=0; i<100; i++){
threadArray[i] = new VolatileTest();
}
for(int i=0; i<100; i++){
threadArray[i].start();
}
}
}
程序运行的结果都不能到达10000,如果加上Synchronized修饰一下就可实现原子性,修改如下:
public void run(){
synchronized(VolatileTest.class){
for(int i=0; i<100; i++){
count++;
}
System.out.println("count="+count);
}
}
关于volatile,暂时就整理这些内容,对于synchronized关键字的一些知识点可以参考我的另外一篇博文synchronized知识点小结