JVM内存模型

(本文谨整理转载,系自己写出来加深印象所用)
之前谈到内存模型的时候总是想到这张图:

JVM内存区域划分

但是这张图其实只是JVM虚拟机中的内存使用分类,即JVM内存在用途上的分类,从概念上这并不是内存模型,参考这篇文章:https://www.jianshu.com/p/76959115d486
即理解上分为这样两类,JVM内存模型干什么的,JVM内存模型的几个分区分别用来做什么事情,第二个方向也就是面试经常问到的问题;

什么是JVM内存模型

Java程序内存的分配是在JVM虚拟机内存分配机制下完成。
Java内存模型(Java Memory Model ,JMM)就是一种符合内存模型规范的,屏蔽了各种硬件和操作系统的访问差异的,保证了Java程序在各种平台下对内存的访问都能保证效果一致的机制及规范。

这里就要梳理一下文章中提到的CPU内存和缓存之间的关系,以及重要概念:内存屏障
首先我们先总结一下内存和缓存的关系:
因为计算机CPU计算性能的快速发展,如果一直IO交互从磁盘读会极大Block程序性能,这也是为什么我们写程序的时候强调少交互,在内存中能完成的事情就不要通过多次交互完成,但是主内存的读写速度也是有瓶颈的,它的读取速度依然跟不上现在的CPU的计算速度,所以大家提出了缓存这么一回事儿,即L1,L2,L3三级缓存结构,读取数据时从1到3依次命中,详见这篇文章多三级缓存结构的解释:https://baijiahao.baidu.com/s?id=1598811284058671259&wfr=spider&for=pc

缓存和内存的结构

但是每个CPU都有这样的缓存结构,多CPU之间如何处理呢,其实是通过MESI协议来完成的,即保证了多CPU下的缓存之间的数据一致性,引用下文https://blog.csdn.net/q865165648/article/details/104095453
MESI协议的作用方式如图:

MESI协议的作用

在硬件层面则是提供了读屏障和写屏障来保证这件事,即内存屏障,其作用是:

由于现代操作系统都是多处理器操作系统,每个处理器都会有自己的缓存,可能存再不同处理器缓存不一致的问题,而且由于操作系统可能存在重排序,导致读取到错误的数据,因此,操作系统提供了一些内存屏障以解决这种问题.
简单来说:
1.在不同CPU执行的不同线程对同一个变量的缓存值不同,为了解决这个问题。
2.用volatile可以解决上面的问题,不同硬件对内存屏障的实现方式不一样。java屏蔽掉这些差异,通过jvm生成内存屏障的指令。

cpu执行指令可能是无序的,它有两个比较重要的作用

a.阻止屏障两侧指令重排序
b.强制把写缓冲区/高速缓存中的脏数据等写回主内存,让缓存中相应的数据失效。
对于读屏障:在指令前插入读屏障,可以让高速缓存中的数据失效,强制从主内存取。

简要言之,JMM是JVM的一种规范,定义了JVM的内存模型。它屏蔽了各种硬件和操作系统的访问差异,不像C语言那样直接访问硬件内存,相对安全很多,它的主要目的是解决由于多线程通过共享内存进行通信时,存在的本地内存数据不一致、编译器会对代码指令重排序、处理器会对代码乱序执行等带来的问题。可以保证并发编程场景中的原子性、可见性和有序性。

JVM内存分类

从线程角度来看:分为线程私有内存,线程共享内存,直接内存,此处引出线程间的数据一致性和数据共享问题

直接内存:
直接内存并不是JVM运行时数据区的一部分,但也会被频繁的使用。在JDK 1.4引入的NIO提供了机遇Channel与Buffer的IO方式,它可以使用Native函数库直接分配堆外的内存,然后使用DirectByteBuffer对象作为这块内存的引用进行操作,这样旧避免了Java堆和Native堆来回赋值数据,因此在一些场景中可以显著提高性能。
本机直接内存的分配不会收到Java堆大小的限制,(即不会遵守-Xms、-Xmx等配置)。但仍然是内存,则肯定还是会收到本机总内存大小+寻址空间的限制,因此扩展时也会出现OutOfMemoryError异常。

从数据区域来分:堆内,堆外,这就比较宽泛了,在JAVA中可以引出堆外缓存中间件的使用,后期我会专门写一篇文章讲述Ehcache 3.7中三级缓存的业务使用介绍;
从内存区域来分,分为五大类:方法区,堆,虚拟机栈,程序计数器,本地方法区,下文依次介绍

方法区(Metaspace)

用于存储已被虚拟机加载的类信息、常量、静态变量,如static修饰的变量加载类的时候就被加载到方法区中。

方法区存储的数据:
类型信息:全限定名、直接超类的全限定名、类的类型还是接口类型、访问修饰符、直接超接口的全限定名的有序列表
字段信息:字段名、字段类型、字段的修饰符
方法信息:方法名、方法返回类型、方法参数的数量和类型(按照顺序)、方法的修饰符
其他信息:除了常量以外的所有类(静态)变量、一个指向ClassLoader的指针、一个指向Class对象的指针、常量池(常量数据以及对其他类型的符号引用)

方法区主要有以下几个特点:
1.方法区是线程安全的,由于所有的线程都共享方法区,所以方法区里的数据访问必须被设计成线程安全的。
2、方法区的大小不必是固定的,JVM可根据应用需要动态调整。
3、方法区也可被垃圾收集,当某个类不在被使用(不可触及)时,JVM将卸载这个类,进行垃圾收集。
HotSpot 虚拟机,很多人愿意把方法区称为“永久代”(Permanent Generation)。本质上两者并不等价,仅仅是因为HotSpot 虚拟机的设计团队选择把GC 分代收集扩展至方法区,或者说使用永久代来实现方法区而已。对于其他虚拟机来说是不存在永久代的概念的。
当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常。

在1.8中通过下面两个参数进行调节:
-XX:MetaspaceSize=XXM
-XX:MaxMetaspaceSize=XXM

针对常量池下图可以很好的概括:


常量池数据

程序计数器

https://www.jianshu.com/p/0ecf020614cb
程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器。在虚拟机的概念模型里(仅是概念模型,各种虚拟机可能会通过一些更高效的方式去实现),字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。
由于Java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(对于多核处理器来说是一个内核)只会执行一条线程中的指令。因此,为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各条线程之间的计数器互不影响,独立存储,我们称这类内存区域为“线程私有”的内存。
如果线程正在执行的是一个Java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是Natvie方法,这个计数器值则为空(Undefined)。
此内存区域是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。
白话文: 程序计数器 可以从计算机原型中理解,它类似与CPU寄存器;用于记录JVM的各自独立线程的执行指令。当然是线程私有。

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

推荐阅读更多精彩内容