今天查看同事写的项目代码,发现有一个变量使用了这个关键字,就上网查 了一下,大概 理解一下。
Java内存模型分为主内存,和工作内存。主内存是所有的线程所共享的,工作内存是每个线程自己有一个,不是共享。
当工作的时候,线程会将主内存的数据备份一份到自己的工作内存中,每次调用数据就会从自己的工作内存中读取。而这样,对于 一些修改的变量别的线程也将无法获取到最新的值。使用volatile关键字,相当于线程每次调用这个变量时都会 告诉这个变量的值是不可靠的,需要去主内存中去对比一下。如果不一样就将主内存中该变量拷贝至该工作内存中更新覆盖。当某个线程对该变量进行了 修改后,强迫该线程把变量值回写到主内存中。这样就保证了线程的可见性。画了一个大概的流程图如下。
volatile 可以保证变量的可见性但是无法保证原子性。之所以使用volatile而不使用synchronized是因为效率会更高,使用synchronized容易导致线程堵塞。
volatile和synchronized的区别:
volatile本质是在告诉jvm当前变量在寄存器(工作内存)中的值是不确定的,需要从主存中读取; synchronized则是锁定当前变量,只有当前线程可以访问该变量,其他线程被阻塞住。
volatile仅能使用在变量级别;synchronized则可以使用在变量、方法、和类级别的
volatile仅能实现变量的修改可见性,不能保证原子性;而synchronized则可以保证变量的修改可见性和原子性
volatile不会造成线程的阻塞;synchronized可能会造成线程的阻塞。
volatile标记的变量不会被编译器优化;synchronized标记的变量可以被编译器优化