作用:volatile关键字的主要作用是使变量在多个线程间可见。强制从公共堆栈中取得变量的值,而不是从线程的私有数据栈中取得变量的值。
使用volatile可以增加实例变量在多个线程之间的可见性,但是缺点是不支持原子性。
volatile和synchronized比较:
1、关键字volatile是线程同步的轻量级实现,所以volatile性能会比synchronized要好,并且volatile只能修饰变量,而synchronized可以修饰方法或者代码块。随着JDK的更新,synchronized关键字的执行效率得到很大的提升,在开发中使用synchronized的比例还是比较大的。
2、多线程访问volatile不会发生阻塞,而synchronized会出现阻塞。
3、volatile关键字能保证数据的可见性,但不能保证原子性;而synchronized可以保证原子性,也可以间接保证可见性,因为它会将私有内存和公共内存中的数据做同步。
4、使用volatile解决的是变量在多个线程之间的可见性,而synchronized解决的是多个线程之间访问资源的同步性。
线程安全包含原子性和可见性两个方面,Java的同步机制都是围绕这两个来确保线程安全的。
关键字volatile提醒线程每次从共享内存中读取变量,而不是从私有内存中读取,这样就保证了同步数据的可见性。但是这里需要注意的是:如果修改实例变量中的数据,如果自加,i++,这样的操作并不是原子操作,是非线程安全的。
表达式 i++ 的操作步骤分解如下:
1、从内存中取出 i 的值;
2、计算 i 的值;
3、将 i 的值写到内存中。
假设在第二步时其他线程也修改了 i 的值,那么就会出现脏读啦。解决方法就是用synchronsized关键字。
volatile本身并不处理数据的原子性,而是强制对数据的读写及时影响到住内存。
变量在内存工作的过程:
1、read和load阶段:从主内存中复制变量到当前线程工作内存;
2、use和assign阶段:执行代码,改变共享变量值;
3、store和write阶段:用工作内存数据刷新主存对应变量的值。
分析:在多线程环境中,use和assign是多次出现的,但是这一操作并不是原子性,也就是说在read和load之后,如果住内存变量发生修改后,线程工作内存中的值由于已经加载,则不会产生相应的变化,也就是私有内存和公共内存中的变量不同步,所以就有了非线程安全的问题。
对于volatile修饰的变量,JVM虚拟机只是保证从住内存加载到线程工作内存的值是最新的。