Java 内存模型- JMM

系列文章
Java 内存模型
Android 系统内存管理机制
Android 性能优化(三)之内存管理
Android 性能优化(四)之内存优化实战


如果本文阐述不太明白,可在文章末尾查看参考资料


JVM - Java Virtual Machine,Java虚拟机
JMM - Java Memory Model,Java内存模型

捋一捋,咱们要讨论的是个角度的JMM?

  1. 涉及 CPU、寄存器、高速缓存、内存 的计算机实际执行的内存模型,姑且叫 动态的内存模型
  2. 仅仅是JVM对内存的物理划分模型,包括程序计数器、线程栈/VM虚拟机栈(线程私有)、本地方法栈、堆(线程共享)、方法区、产量池等,姑且叫 静态的内存模型

上面是以两个不同的维度来理解JMM。

认识 Java 线程安全,必须了解两个主要的点:

  1. Java的内存模型
  2. Java的线程同步机制,很大程度上都是基于内存模型而设定的。

关键字

语言级的内存模型,硬件级的内存模型,JSR-133,JMM抽象模型,并发,多线程,线程安全,原子性,有序性,可见性,数据依赖,重排序(编译器,处理器指令级,处理器系统缓存),内存屏障,happens-before语义,as-if-serial语义,数据竞争,顺序一致性,语法同步原语(lock / volatile / final),临界区,锁竞争,就绪队列,阻塞队列

看着很多,混个眼熟就行

其他随意补充:
transient 短暂的,不被序列化
volatile 不稳定的,保证内存可见性

知识点补充

可选择性忽略

  1. 进程和线程的区别
  2. 现在的计算机,cpu在计算的时候,并不总是从内存读取数据,它的数据读取顺序优先级 是:寄存器-高速缓存-内存。
  3. 线程耗费的是CPU,线程计算的时候,原始的数据来自内存,在计算过程中,有些数据可能被频繁读取,这些数据被存储在寄存器 和高速缓存中,当线程计算完后,这些缓存的数据在适当的时候应该写回内存。当个多个线程同时读写某个内存数据时,就会产生多线程并发问题,涉及到三个特 性:原子性,有序性,可见性。
  4. JVM是一个虚拟的计算机,它也会面临多线程并发问题,java程序运行在java虚拟机平台上,java程序员不可能直接去控制底层线程对寄存器高速缓存内存之间的同 步,那么java从语法层面,应该给开发人员提供一种解决方案,这个方案就是诸如 synchronized, volatile,锁机制(如同步块,就绪队 列,阻塞队列)等等。这些方案只是语法层面的,但我们要从本质上去理解它,不能仅仅知道一个 synchronized 可以保证同步就完了。
  5. Java 程序执行的过程:Java源程序 *.java -> Java编译器-> 字节码文件 *.class -> Java解释器 -> 机器指令运行 -> 内存 -> 总线 -> 高速缓存 -> 寄存器 -> CPU处理

相关图片

先抛出来几张图随意感受一下

1. Java内存模型的*抽象*示意图;来源:深入理解 Java 内存模型(一)

注意1:这张图是抽象示意图
注意2:线程的“工作内存”或“本地内存”或“working memory”,是一个抽象的概念,本质是 CPU的寄存器和高速缓存的抽象描述,由于数据不同步导致的一种线程拥有工作内存的概念,实际上不会分配工作内存的空间。

2. 总线;来源:深入理解 Java 内存模型(三)
3. 来源:CPU高速缓存行与内存关系 及并发MESI 协议
4. JVM运行时数据区;来源:JVM 答疑解惑
5. 整体执行简图

堆内存线程共享,栈不共享不存在可见性问题

JMM到底是什么?

Java 线程之间的通信由 Java 内存模型(本文简称为 JMM)控制,JMM 决定一个线程对共享变量的写入何时对另一个线程可见。

从抽象的角度来看,JMM 定义了线程和主内存之间的抽象关系:线程之间的共享变量存储在主内存(main memory)中,每个线程都有一个私有的本地内存(local memory),本地内存中存储了该线程以读 / 写共享变量的副本。

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

屏蔽了各种硬件和操作系统的内存访问差异,以实现让Java程序在各种平台下能达到一致的内存访问效果。

对Java开发人员来说,抽象了工作内存的概念,并提供了保证线程安全的同步原语(lock/volatile/final);

浅谈Java内存模型

并发编程模型有 共享内存模型 和消息传递模型两种。JVM采用共享内存模型。

不同的平台,内存模型是不一样的,但JVM的内存模型规范是统一的。

线程安全:Java的多线程并发最终都会反应在Java的内存模型上,而JMM 就要控制多个线程对多个资源的有序访问或修改。

Java的内存模型要解决的两个问题:可见性和有序性(包括原子性)。

JVM定义了自己的内存模型,屏蔽了底层平台内存管理细节,对于Java开发人员,要清楚在 JVM 内存模型的基础上,解决多线程的可见性和有序性。

可见性和有序性

JVM规范定义了线程对主存的操作指 令:read,load,use,assign(分配指定),store,write。

何为可见性?

多个线程之间是不能互相传递数据通信的,他们的沟通只能通过共享变量来进行。(共享内存模型)

Java 内存模型(JMM)规定了 JVM 有主内存,主内存是由多个线程共享的。当new一个对象的时候,也是被分配在主内存中,每个线程都有自己的工作内存,工作内存存储了主存的某些对象的副本,当然线程的工作内存大小是有限制 的。当线程操作某个对象时,执行顺序如下:

  1. 从主存复制变量到当前工作内存 (read and load)
  2. 执行代码,改变共享变量值 (use and assign)
  3. 用工作内存数据刷新主存相关内容 (store and write)

当一个共享变量在多个线程的工作内存中都有副本时,如果一个线程修改了这个共享 变量,那么其他线程应该能够看到这个被修改后的值,这就是多线程的可见性问题。

何为有序性?

线程在引用变量时不能直接从主内存中引用,如果线程工作内存中没有该变量,则会从主内存中拷贝一个副本到工作内存中,这个过程为read-load,完 成后线程会引用该副本。当同一线程再度引用该字段时,有可能重新从主存中获取变量副本(read-load-use),也有可能直接引用原来的副本 (use),也就是说 read,load,use顺序可以由JVM实现系统决定

线程不能直接为主存中中字段赋值,它会将值指定给工作内存中的变量副本(assign),完成后这个变量副本会同步到主存储区(store- write),至于何时同步过去,根据JVM实现系统决定

有该字段,则会从主内存中将该字段赋值到工作内存中,这个过程为read-load,完成后线程会引用该变量副本

当同一线程多次重复对字段赋值时,线程有可能只对工作内存中的副本进行赋值,只到最后一次赋值后才同步到主存储区,所以assign,store,weite顺序可以由JVM实现系统决定。

以下说明沿袭“working memory”的说法,不牵扯到太多底层细节,从语法层面理解java的线程同步,知道各个关键字的使用场景

语义保证线程安全

synchronized 解决多线程的执行有序性和内存可见性问题,保证同步块是互斥的。
volatile 只解决多线程的内存可见性问题(轻量级的同步,不保证执行有序性)。

final 不可变,只能在构造方法中赋值一次。

JSR-133 之后对 volatile和 final 增强了语义,保证内存可见性

synchronized

synchronized 标示了临界区或互斥区

方法锁
public synchronized void add(int num),锁就是这个方法所在的对象
public static synchronized void add(int num),锁就是这个方法所在的class

理论上,每个对象都可以做为锁,但一个对象做为锁时,应该被多个线程共享,这样才显得有意义,在并发环境下,一个没有共享的对象作为锁是没有意义的。

每个锁对象都有两个队列,一个是就绪队列,一个是阻塞队列

就绪队列存储了 将要 获得锁的线程,阻塞队列存储了被阻塞的线程(wait),当一个被线程被唤醒 (notify)后,才会进入到就绪队列,等待cpu的调度。

一个线程执行临界区代码过程如下:

  1. 获得同步锁
  2. 清空工作内存
  3. 从主存拷贝变量副本到工作内存
  4. 对这些变量计算
  5. 将变量从工作内存写回到主存
  6. 释放锁
    可见,synchronized既保证了多线程的并发有序性,又保证了多线程的内存可见性。

volatile

轻量级的同步,只能保证多线程的内存可见性,不能保证多线 程的执行有序性。

任何被 volatile 修饰的变量,都不拷贝副本到工作内存,任何 修改都及时写在主存。因此对于 volatile 修饰的变量的修改,所有线程马上就能看到,但是volatile不能保证对变量的修改是有序的。

要使 volatile 变量提供理想的线程安全,必须同时满足下面两个条件:
1)对 变量的写操作不依赖于当前值。
2)该变量没有包含在具有其他变量的不变式中

适合场景:直接进行赋值操作,开销非常小

参考资料

JSR-133 官方文档
线程安全总结(一)
线程安全总结(二)
深入理解java内存模型系列文章
深入理解Java内存模型
CPU高速缓存行与内存关系 及并发MESI 协议
请问CPU,内核,寄存器,缓存,RAM,ROM的作用和他们之间的联系?
JVM 答疑解惑

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容