1 volatile的定义
Java语言规范第3版中对volatile的定义如下:Java编程语言允许线程访问共享变量,为了确保共享变量能被准确和一致地更新,线程应该确保通过排他锁单独获得这个变量。Java语言提供了volatile,在某些情况下比锁要更加方便。如果一个字段被声明成volatile,Java线程内存模型确保所有线程看到这个变量的值是一致的。
2 volatile是如何来保证可见性的呢?
让我们在X86处理器下通过工具获取JIT编译器生成的汇编指令来查看对volatile进行写操作时,CPU会做什么事情。
Java代码如下
转变成汇编代码,如下。
有volatile变量修饰的共享变量进行写操作的时候会多出第二行汇编代码,通过查IA-32架构软件开发者手册可知,Lock前缀的指令在多核处理器下会引发了两件事情:
- 将当前处理器缓存行的数据写回到系统内存。
- 这个写回内存的操作会使在其他CPU里缓存了该内存地址的数据无效(与普通变量不同的地方)
为了提高处理速度,处理器不直接和内存进行通信,而是先将系统内存的数据读到内部缓存(L1,L2或其他)后再进行操作,但操作完不知道何时会写到内存。如果对声明了volatile的变量进行写操作,JVM就会向处理器发送一条Lock前缀的指令,将这个变量所在缓存行的数据写回到系统内存。但是,就算写回到内存,如果其他处理器缓存的值还是旧的,再执行计算操作就会有问题。所以,在多处理器下,为了保证各个处理器的缓存是一致的,就会实现缓存一致性协议,每个处理器通过嗅探在总线上传播的数据来检查自己缓存的值是不是过期了,当处理器发现自己缓存行对应的内存地址被修改,就会将当前处理器的缓存行设置成无效状态,当处理器对这个数据进行修改操作的时候,会重新从系统内存中把数据读到处理器缓存里。
3 volatile的作用
- 保证被修饰变量对其他线程的可见性(并不能保证并发安全)
- 禁止指令重排
4 volatile不能保证并发的安全性
package basicKnowledge.jvm.volatileTest;
/**
* @基本功能:测试volatile关键字的线程安全性
* @program:summary
* @author:peicc
* @create:2019-08-28 16:04:13
**/
public class TestVolatile{
public static volatile int race = 0;
public static void increase(){
race++;
}
public static void main(String[] args){
Thread[] threads = new Thread[10];
for(int i = 0; i < 10; i++){
//每个线程对t进行1000次加1的操作
threads[i]=new Thread(new Runnable(){
@Override
public void run(){
for(int j = 0; j < 1000; j++){
increase();
}
}
});
threads[i].start();
}
//等待所有累加线程都结束
while(Thread.activeCount() > 1){
Thread.yield();
}
//打印t的值
System.out.println(race);
}
}
输出结果并不是10000,少于10000并且每次都不一样
结果分析
当线程1读取了race的值(比如0),此时线程2也读取了race的值(也为0),然后线程1对race的值加1,但并未同步到主内存中,而线程2在对race执行加1的动作时,也是在0的基础上加的。这样当线程1,2将值再同步回主内存时,race的值只为1。