在聊volatile的原理前,有必要先了解下内存屏障是什么。
内存屏障
它是一个CPU指令。通俗来讲,它能确保一些特定操作执行的顺序,并且可以影响一些数据的可见性(可能是某些指令执行后的结果)。
- 编译器和CPU可以在保证输出结果一样的情况下对指令重排序,使性能得到优化。插入一个内存屏障,相当于告诉CPU和编译器先于这个命令的必须先执行,后于这个命令的必须后执行。
- 内存屏障另一个作用是强制更新一次不同CPU的缓存。例如,一个写屏障会把这个屏障前写入的数据刷新到缓存,这样任何试图读取该数据的线程将得到最新值,而不用考虑到底是被哪个cpu核心或者哪颗CPU执行的;
volatile与内存屏障的关系
在Java中,如果你的字段是volatile
,Java内存模型将在写操作后插入一个写屏障指令,在读操作前插入一个读屏障指令。
这意味着如果你对一个volatile
字段进行写操作:
- 一旦你完成写入,任何访问这个字段的线程将获得最新的值。
- 在你写入前,会保证所以之前的事已经发生,并且任何更新过的数据值也是可见的,因为内存屏障会把之前的写入值都刷新到缓存中。
这样就保证了volatile
变量的可见性和赋值操作不会被重排序。
对性能的影响
内存屏障作为另一个CPU级的指令,没有锁那样大的开销。内核并没有在多个线程间干涉和调度。但凡事都是有代价的。内存屏障的确是有开销的——编译器/cpu不能重排序指令,导致不可以尽可能地高效利用CPU,另外刷新缓存亦会有开销。
总结
内存屏障是CPU指令,它允许你对数据什么时候对其他进程可见作出假设。在Java里,你使用volatile
关键字来实现内存屏障。使用volatile
意味着你不用被迫选择加锁,并且还能让你获得性能的提升。
但是,你需要对你的设计进行一些更细致的思考,特别是你对volatile
字段的使用有多频繁,以及对它们的读写有多频繁。