JVM内存管理—运行时内存区域

JVM 有两种类型的线程:守护线程,非守护线程,只有所有非守护线程都结束之后,JVM才会结束运行,退出。守护线程如GC,非守护线程如main。

操作系统内存与JVM内存的联系与区别

  1. 操作系统分为,栈由操作系统管理,会有操作系统进行自动回收,堆由用户进行分配使用
  2. JVM内存使用的操作系统的堆,以防JVM分配的内存被操作系统回收
  3. JVM本地方法栈指的是操作系统的栈
  4. 操作系统的PC寄存器,是计算机上的存储硬件,与内存条一样的硬件,但是寄存区位于CPU内,被称为Cache,用于加快数据访问速度;内存是外挂在CPU的数据总线上的
  5. JVM PC寄存器位于操作系统的堆

图片来源

操作系统与JVM内存模型

JVM规范中的内存空间

1- JVM PC寄存器

PC寄存器配合字节码解释器,选取下一条字节码指令来解释执行,线程私有,每个线程都有一个PC寄存器。为了确保切换线程后能恢复到原来进程正确的执行位置。如果执行的不是Java方法,而是本地方法Native Method,这个计数器值为空(Undefined),如果是Java方法则保存的是正在执行的虚拟机字节码指令地址,可以看做是当前线程所执行的字节码的行号指示器,会自增取下一条字节码指令的地址。这是JVM 规范的唯一没有OutMemoryError的内存区域。

2- Java 虚拟机栈
  • 启动一个线程时,JVM就会给这个线程分配一个栈,所以栈的生命周期是和线程一样的。
  • Java方法执行的内存模型:每个Java方法的执行都会创建一个栈帧(Stack Frame),方法的调用到执行完毕(正常退出,或者异常退出)对应着一个栈帧的入栈出栈的过程
  • 一个栈帧包括:局部变量表、操作数栈、动态链接、方法出口等。

栈帧结构:

  1. 局部变量表:存放的是编译期间可知的基本数据类型,和对象的引用,比如方法的参数变量,还有方法内的局部变量;需要注意的是如果这个方法是普通的方法,那么还会有自己的本身对象的一个引用(this,索引为0),如果是静态方法没有这个自身对象的引用。一个单位局部变量空间为32位64位长度的long和double数据类型会占用两个局部变量空间。变量通过声明的顺序的索引来进行访问的,
  2. 操作数栈(Operand Stack):是一个存储中间变量结果的栈结构,只能通过入栈和出栈来进行访问。Java虚拟机指令是主要是通过操作数栈来获取操作数(Operand)的,而不是寄存器。
int a  = 100;  
int b = 98;
// iload_0,iload_1,iadd,istore_2
//iload指令是指将局部变量表中一个int变量加载到操作栈中
//iadd指令弹出操作数栈的两个变量,进行加法运算,然后将结果压入栈中
//istore指令是指将操作栈一个数值存储到局部变量表中
int c = a+b;
操作数栈
  1. 动态链接(待续)
  2. 方法出口(待续)
  3. 通常程序员所说的指的就是虚拟机的局部变量表
  4. 两种异常:StackOverflowError(栈空间溢出),当线程请求的栈深度大于虚拟机所允许的深度,将会抛出此异常;OutMemoryError(栈空间拓展内存溢出),当虚拟机栈支持动态拓展时,如果在扩展时无法申请到足够的内存时,就会抛出此异常。
3- 本地方法栈(Native Method)
  1. 本地方法栈虚拟机栈的作用是一样的,只是服务的对象不一样,虚拟机方法栈是为虚拟机执行java方法(字节码)服务的,而本地方法栈是为虚拟机执行Native方法服务的。
  2. Sun HotSpot 直接将本地方法栈虚拟机栈 合二为一
  3. 两种异常:和虚拟机栈一样
4- Java堆(Java Heap)
  1. 所有线程共享的内存区域,与虚拟机同生命周期
  2. 主要任务是存储对象实例,基本上所有对象实例数组都在Heap上分配空间,但是也不这么绝对,因为编译器优化。
  3. Java堆是一块很大的内存区域,为了加速GC回收的效率,把这个堆有按照不同粒度进行细分
    • 按代划分 新生代,老年代;再细分为:Eden空间、From Survivor空间、To Survivor 空间。
    • 从内存分配的角度: TLAB(Thread Local Allocation Buffer):线程共享Heap中可以多个线程私有的分配缓存。
    • GC的主要回收区域
    • 物理存储上不一定连续,只要逻辑上连续即可;堆空间可以拓展(通过-Xmx,-Xms来控制)。
    • 异常:OutMemoryError 当堆中没有可用内存来存储对象实例
5- 方法区(Method Area)
  1. 所有线程共享的内存区域,与虚拟机同生命周期

  2. 存储已被虚拟机加载的类信息、常量、静态变量、编译后的代码

  3. 按照代划分,方法区在HotSpot中被划分为永久代(Permanent Generation)

  4. 不需要物理连续的存储空间,可拓展(通过 -XX:MaxPermSize,-XX:MinPermSize)

  5. 永久代并不永久,GC会对常量池的回收,以及类型的卸载

  6. 异常:OutMemoryError,当方法区无法满足内存分配时。

  7. 运行时常量池(Runtime Constant Pool)

    • 是方法区的一部分,Class文件中除了有版本、字段、方法、接口等描述信息,还会有一项信息是常量池,用于存放编译期间生成的各种字面量符号引用以及翻译后的直接引用,这部分内容将在类加载后存放在方法区的运行时常量池中。
    • 运行时常量池中的常量,不一定来源于一开始加载的Class文件(编译期间产生常量),也可以在运行时将新的常量放入常量池中,这是运行时常量池动态性的体现。

JVM规范外的内存空间 -- 直接内存(Direct Memory)

  1. 不是虚拟机内存模型以及数据区的一部分,但是频繁使用,特别是NIO中,基于通道(channel)缓冲区(Buffer)的I/O方式
  2. 通过Native函数库直接分配堆外内存,能在一些场景下显著提高性能,因为避免了在Java堆和Native堆中来回复制数据
  3. 不会受到Java堆的大小影响,主要取决于本机的内存,以及处理器的寻址空间
  4. OutMemoryError:虚拟机参数设置时,将虚拟机内存设置超出物理内存或者操作系统的限制时,导致动态拓展时导致内存溢出。

java7内存布局的变化

从JDK7开始永久代的移除工作,但永久代仍然存在于JDK7,并没有完全的移除。

  1. 符号引用(Symbols)转移到了native heap(java堆外);
  2. 字面量(interned strings)和类的静态变量(class statics)转移到了java heap。
  3. Class元数据还在方法区上

java8 元空间

元空间的本质和永久代类似,都是对JVM规范中方法区的实现,用于存放Class元数据。不过元空间与永久代之间最大的区别在于:元空间并不在虚拟机中,而是使用本地内存,这部分内存区域间接被GC管理。因此,默认情况下,元空间的大小仅受本地内存限制,但是实际使用上如果不设置大小,可能耗尽系统内存。


参考链接:
TLAB与PLAB
http://www.jianshu.com/p/2343f2c0ecc4
http://www.jianshu.com/p/cd85098cca39
NIO-Buffer
http://www.jianshu.com/p/fb832bc2cc32
Java 8 元数据空间
http://blog.csdn.net/zhushuai1221/article/details/52122880

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

推荐阅读更多精彩内容