欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

JMM内存模型详解(一)

程序员文章站 2023-11-13 09:31:34
本文开始死磕JMM(Java内存模型)由于知识点较多,分来写 该文为JMM第一篇 技术往往是枯燥的,本文文字较多 1. JMM是什么? 其实JMM很好理解,我简单的解释一下,在Java多线程中我们经常会涉及到两个概念就是线程之间是如何通信和线程之间的同步,那什么是线程之间的通信呢,其实就是两个线程之 ......

本文开始死磕jmm(java内存模型)由于知识点较多,分来写

该文为jmm第一篇

技术往往是枯燥的,本文文字较多

1. jmm是什么?

其实jmm很好理解,我简单的解释一下,在java多线程中我们经常会涉及到两个概念就是线程之间是如何通信和线程之间的同步,那什么是线程之间的通信呢,其实就是两个线程之间互相交换信息线程之间通信的方式共有两种:一种就是共享内存,和消息传递。在共享内存中的并发模型中线程是通过读取主内存的共享信息来进行隐性通信的。在消息传递通信中线程之间没有公共的状态,只能通过发送消息来进行显性通信。然而这只是线程通信,那么同步呢,同步就是在多线程的情况下有顺序的去执行。在共享内存中同步时显式进行的,在代码中我们必须要去指定方法需要同步执行比如说加同步锁等。在消息传递的并发模型中发送消息必须是在消接收之前,所以同步时隐式的。

2.为什么要涉及到线程并发通信

java内存模型其实可以说是java并发内存模型,在java中是采用的共享内存模型的方式,所以java线程之间的通信是隐式进行的,对我们是完全透明的,如果你不了解通信机制的话会产生各种线程可见性的问题。其实在java中所有的静态域,域和数组元素都存在堆内存中,堆内存在线程中是共享的一般我们都称之为共享变量,局部变量,方法定义参数和异常处理参数不会在线程*享,所以不会存在线程可见性的问题。上面我就说过线程之间的通信是由jmm来进行控制的,jmm来决定了一个线程操作了共享变量后如何对另一个线程可见。从上面所说的概念来看的话,jmm定义了线程与主内存的关系。

3.jmm规定

其实这样做的原因就是java是跨平台语言,在个操作系统中内存都有一定的差异性,这样久造成了并发不一致,所以jmm的作用就是用来屏蔽掉不同操作系统中的内存差异性来保持并发的一致性。同时jmm也规范了jvm如何与计算机内存进行交互。简单的来说jmm就是java自己的一套协议来屏蔽掉各种硬件和操作系统的内存访问差异,实现平台一致性达到最终的"一次编写,到处运行",说了这么多,jmm到底是怎么控制的呢?然后如何通信的呢?我们继续往下看。

4.模型

jmm是一个抽象的概念,并不是真实的存在,它涵盖了缓冲区,寄存器以及其他硬件和编译器优化。

java内存模型抽象图如下:

JMM内存模型详解(一)
从上图可以看出每个线程都有一个本地内存,如果线程想要通信的话要执行一下步骤:

  • a线程先把本地内存的值写入主内存
  • b线程从主内存中去读取出a线程写的值

再看下面的这个图,表示了a如何向b发送消息

JMM内存模型详解(一)

假设这时候有一个共享变量x默认值都是为0,那么线程a把x的值修改为1,这时候如何才能同步到b线程呢。

如果a线程把x修改成1之后,a线程会把x从a的本地内存中写入到主内存中,这样的话主内存的x就等于1了,这时候b线程就会去读取主内存的x变量,存入b的本地内存中,这样b线程的x变量值也就会变成了1。这样对吗。那现在如何通信我是知道了关键它究竟是如何来实现的,就是如何来实现通信的呢?

5.通信

上面所说的步骤其实就是实现了线程之间的通信,但是不要以为线程之间的通信就是这么简单的,其实在java中jmm内存模型定义了八种操作来实现同步的细节。

  • read 读取,作用于主内存把变量从主内存中读取到本本地内存。
  • load 加载,主要作用本地内存,把从主内存中读取的变量加载到本地内存的变量副本中
  • use 使用,主要作用本地内存,把工作内存中的一个变量值传递给执行引擎,每当虚拟机遇到一个需要使用变量的值的字节码指令时将会执行这个操作。、
  • assign 赋值 作用于工作内存的变量,它把一个从执行引擎接收到的值赋值给工作内存的变量,每当虚拟机遇到一个给变量赋值的字节码指令时执行这个操作。
  • store 存储 作用于工作内存的变量,把工作内存中的一个变量的值传送到主内存中,以便随后的write的操作。
  • write 写入 作用于主内存的变量,它把store操作从工作内存中一个变量的值传送到主内存的变量中。
  • lock 锁定 :作用于主内存的变量,把一个变量标识为一条线程独占状态。
  • unlock 解锁:作用于主内存变量,把一个处于锁定状态的变量释放出来,释放后的变量才可以被其他线程锁定。

所以看似简单的通信其实是这八种状态来实现的。

同时在java内存模型中明确规定了要执行这些操作需要满足以下规则:

  • 不允许read和load、store和write的操作单独出现。
  • 不允许一个线程丢弃它的最近assign的操作,即变量在工作内存中改变了之后必须同步到主内存中。
  • 不允许一个线程无原因地(没有发生过任何assign操作)把数据从工作内存同步回主内存中。
  • 一个新的变量只能在主内存中诞生,不允许在工作内存中直接使用一个未被初始化(load或assign)的变量。即就是对一个变量实施use和store操作之前,必须先执行过了assign和load操作。
  • 一个变量在同一时刻只允许一条线程对其进行lock操作,lock和unlock必须成对出现
  • 如果对一个变量执行lock操作,将会清空工作内存中此变量的值,在执行引擎使用这个变量前需要重新执行load或assign操作初始化变量的值
  • 如果一个变量事先没有被lock操作锁定,则不允许对它执行unlock操作;也不允许去unlock一个被其他线程锁定的变量。
  • 对一个变量执行unlock操作之前,必须先把此变量同步到主内存中(执行store和write操作)。

所以上面说的操作要严格执行。

目前写了这么多,下文预告:

linkedhashmap源码分析

参考资料《深入java内存模型》