CPU缓存
Java指令操作CPU缓存时,JVM为提高程序运行效率,会对指令进行重排序,比如写缓存的效率慢于读缓存的效率,在同时存在写和读操作指令时,读操作的指令先执行。但要遵守as-if-serial语义。
as-if-serial语义
即不管怎么重排序(编译器和处理器为了提高并行度),(单线程)程序的执行结果不能被改变。编译器,runtime 和处理器都必须遵守as-if-serial语义。为了遵守as-if-serial语义,编译器和处理器不会对存在数据依赖关系的操作做重排序,因为这种重排序会改变执行结果。但是,如果操作之间不存在数据依赖关系,这些操作可能被编译器和处理器重排序。
在实际情况中,几乎没有单线程的服务器。所以在多CPU多线程的状况下,单靠as-if-serial语义无法保证重排序后对程序的执行结果没有影响,CPU厂商在在进行设计时对CPU缓存的操作增加了两个内存屏障指令,写内存屏障和读内存屏障,来解决这个问题。
内存屏障
写内存屏障
在指令后插入写内存屏障,能让写入CPU缓存的最新数据更新写入主内存,让其他线程可见。强制写入主内存,这种显示调用,CPU就不会因为性能考虑而去进行指令重排序。
读内存屏障
在读指令前插入读内存屏障,可以强制从主内存加载数据。强制读取主内存内容。让CPU缓存与主内存保持一致,避免了缓存导致的一致性问题。