1. volatile是什么?
volatile是Java提供的一种轻量级的同步机制。与synchronized修饰方法、代码块不同,volatile只用来修饰变量。并且与synchronized、ReentrantLock等重量级锁不同的是,volatile更轻量级,因为它不会引起线程上下文的切换和调度。
2. volatile的作用
说volatile作用之前,先说一下并发编程的三大特性:原子性、可见性和有序性。
-
原子性
即一个或者多个操作作为一个整体,要么全部执行,要么都不执行,并且操作在执行过程中不会被线程调度机制打断;而且这种操作一旦开始,就一直运行到结束,中间不会有任何上下文切换。
-
可见性
可见性是指当多个线程访问同一个变量时,一个线程修改了这个变量的值,其他线程能够立即看得到修改的值。
-
有序性
为了提高程序的执行效率,编译器会对编译后的指令进行重排序,即代码的编写顺序不一定就是代码的执行顺序。
并发编程中只有同时满足这三大特性,才能保证程序正确的执行。而volatile的只保证了可见性和有序性,不保证原子性。
volatile的作用只有两个:
-
保证内存的可见性
-
禁止JVM内存重排序(保证有序性)
在并发多线程情况下,为什么会有可见性问题?如果不做控制,为什么一个线程修改了共享变量的值,其他线程不能立即看到?这就需要聊到JMM(Java内存模型,Java Memory Model)。
3. JMM是什么
JMM(Java内存模型,Java Memory Model)定义程序访问变量的规范,为了屏蔽不同操作系统之间的差异。
由于Java共享变量是存储在主内存中,而Java线程无法直接访问主内存中数据,只能把主内存中的数据读到本地内存(相当于拷贝一份副本),修改完本地内存的数据,再写回主内存。而此时另一个线程也把主内存的数据拷贝到自己私有的本地内存中,虽然线程1已经修改