Javaer都知道,我们在编译器上面编写的Java代码经过编译后会形成字节码,然后由类加载器加载到JVM中,JVM在执行字节码时,将它们转换成一条条的汇编指令,最终由CPU的寄存器来运行,在CPU执行这些汇编的过程中需要读取数据或者写入数据,但CPU能读取的数据只能来自计算机中的内存,随着科技的发展,像Intel的部分CPU频率特别是睿频后已经到达了4.5GHZ了,但内存发展就比较缓慢,比如顶级的内存就3600MHZ左右,因此就造成了CPU的处理速度已经远远超过了内存的访问速度,正常情况都是千倍的速度差距。

  CPU缓存模型

  因为速度差距过大的原因,如果还是采用CPU直接读取内存上面的数据,就会导致CPU资源严重的浪费!于是那些生产CPU的科技公司就设计出了,在CPU和内存之间增加一层缓存的方案,刚才刻意到京东查了一下,现在的CPU基本都是三级缓存了,L1 ,L2 ,L3 缓存,

  

深入底层了解Java并发机制之CPU缓存模型_第1张图片


  CPU缓存模型

  

深入底层了解Java并发机制之CPU缓存模型_第2张图片


  CPU缓存和内存访问速度对比图

  通过这两张图,我们就可以更加直观地感受到CPU缓存和内存在访问上面的速度的差距了,至于CPU核心的计算速度,和他们相比又是另一个级别的差距了。

  那么在有了CPU缓存之后,我们就可以在程序运行的过程中,先从内存拷贝一份数据到CPU缓存中,然后CPU计算都操作缓存里的数据,等执行完成的时候,再把缓存中的数据更新到内存里,从而增加CPU的使用效率。

  

深入底层了解Java并发机制之CPU缓存模型_第3张图片


  CPU借助缓存和内存进行数据交互

  在引入CPU缓存之后,主了提高CPU的使用效率之外,还带来了一个数据不一致的问题。比如i++这一个操作,在引入了CPU缓存之后,他具体的情况是这样的:

深入底层了解Java并发机制之CPU缓存模型_第4张图片

  对于单线程来说,这完成不会有什么问题,但是对于多线程来说,就会出现错误了,因为每个线程都有自己的工作空间。比如,现在有线程A和线程B同时对i执行i++操作,我们假设i一开始为0,我们期望最后的结果是2,但是最后的结果可能1:比如:

深入底层了解Java并发机制之CPU缓存模型_第5张图片

  出现这种情况的原因也是很简单的,比如多个CPU核心都从内存拷贝了一份数据到各自的缓存当中,然后直接拿缓存中的数据来执行+1操作,最后再把数据刷新内存,于是就造成了这个问题。由于Demo过于简单,我就不给出来了。下面我们回顾一下历史,看看这个问题是怎么被解决的,其实解决这个问题的方案有两种:

  但是这种方案明显效率是比较低的,于是就提出了第二方案:

  通过缓存一致性协议来解决数据不一致的问题,即CPU在操作CPU缓存中的数据时,

  如果发现它是一个共享变量(其他CPU也缓存了一个副本),那么他会进行以下的两种操作:

  (1) 读操作,只会将数据单纯读到寄存器,不做额外处理

  (2) 写操作,发出一个信号告诉其他CPU核心,你缓存的数据已经无效啦,让其他CPU在读取共享变量时,不得不重新去内存