深入理解JMM

开始之前,我们先来了解一下cpu和内存的关系

       我们先对图进行介绍一下,在CPU内部有一组CPU寄存器,也就是CPU的储存器。CPU操作寄存器的速度要比操作计算机主存快的多,当计算机很快执行完任务时,当然是不能一直等待吧,所以引入了cpu的缓存,一级缓存,二级缓存,三级缓存。存在于主存和CPU寄存器之间,CPU操作CPU缓存的速度快于主存但慢于CPU寄存器。某些CPU可能有多个缓存层(一级缓存和二级缓存)。计算机的主存也称作RAM,所有的CPU都能够访问主存,而且主存比上面提到的缓存和寄存器大很多。

       当一个cpu需要访问主存的时候,它会先去读取主存的部分数据到cpu缓存中,然后再读取cpu缓存到寄存器,当cpu需要写入数据到主存时,是一个逆向的过程,先是寄存器刷新数据到cpu缓存中,接着再将cpu缓存的数据刷新到主存,完成写入的过程。接下来就开始了解Java内存模型了。

Java内存模型

       Java内存模型即Java Memory Model,简称JMM。而JMM定义了Java 虚拟机(JVM)在计算机内存(RAM)中的工作方式。JVM是整个计算机虚拟模型,所以JMM是隶属于JVM的。

JMM本身是一种抽象的概念,并不真实存在,它描述的是一组规则或规范,通过这组规范定义了程序中各个变量(包括实例字段,静态字段和构成数组对象的元素)的访问方式。由于JVM运行程序的实体是线程,而每个线程创建时JVM都会为其创建一个工作内存(有些地方称为栈空间),用于存储线程私有的数据,而Java内存模型中规定所有变量都存储在主内存,主内存是共享内存区域,所有线程都可以访问,但线程对变量的操作(读取赋值等)必须在工作内存中进行,首先要将变量从主内存拷贝的自己的工作内存空间,然后对变量进行操作,操作完成后再将变量写回主内存,不能直接操作主内存中的变量,工作内存中存储着主内存中的变量副本拷贝,前面说过,工作内存是每个线程的私有数据区域,因此不同的线程间无法访问对方的工作内存,线程间的通信(传值)必须通过主内存来完成,其简要访问过程如下图

由上图可以看到,线程A和B要想修改共享变量的值,要先从主内存中获取共享变量的副本,然后放到自己的工作内存中(即本地内存),再在自己的工作内存中对变量进行修改,重新刷入到主内存中,这样就完成了变量的修改过程。

举个例子吧

我们对比cpu和内存的关系图和java内存模型时,个人的理解:我们现在的电脑基本都是4核心以上,比如四核八线程,还有更多,支持更多核心和线程,所以cpu先将主存中a=1放到cpu缓存中,当需要对a的值修改时,将进入寄存器,对a的值进行修改,然后重新写入主存中。参考这位老哥的文章:

多线程和cpu的关系

线程A和B先是将a=1读取到自己的工作内存中,线程A进行a++操作后,再将a=2刷入主内存,这时,主内存中a的值变为2,而线程B中a的值仍是1,如果线程2的值想要变成2的话,则需要对a变量加上volatile关键字,

下一节我们来一起了解、分析、深入volatile的作用。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值