【JVM】Java内存区域详解

本文将介绍了 Java 虚拟机内存的各个区域以及这些区域的作用、服务对象和其中可能出现的异常等。

JVM 运行时数据区域

Java 虚拟机在执行 Java 程序的过程中会把它所管理的内存划分为五个不同的数据区域(如图所示)。

image
  • 红色边框的是由所有线程共享的数据区
  • 蓝色边框的是线程隔离的数据区

除了程序计数器之外,其他四个区域都可能会出现 OutOfMemoryError 异常。

程序计数器

  • 程序计数器是一块较小的内存空间

  • 是当前线程所执行的字节码的行号显示器

  • 字节码解释器就是通过改变这个计数器的值来选取下一条需要执行的字节码指令的

  • 执行 Java 方法和执行 Native 方法的区别:

    • 执行 Java 方法时,计数器记录虚拟机正在执行的字节码指令的地址
    • 执行 Native 方法时,无记录,也就是计数器的值为空(Undefined)

程序计数器是唯一一个不会出现 OOM 的区域。

Java 虚拟机栈

  • 每个 Java 方法被执行的时候,Java 虚拟机都会同步创建一个栈帧用于存储局部变量表、操作数栈、动态连接、方法出口等信息

  • 每个方法被调用一直到执行完毕的过程,都对应着一个栈帧在 Java 虚拟机栈中从入栈到出栈的过程

  • Java 虚拟机栈服务于 Java 方法

  • 可能出现的异常:

    • StackOverflowError:线程请求的栈深度 大于 Java 虚拟机所允许的深度时
    • OutOfMemoryError:在 Java 虚拟机栈容量可动态扩展的情况下,当栈扩展时无法申请到足够的内存时
  • 虚拟机参数设置:-Xss

本地方法栈

  • 与 Java 虚拟机栈发挥的作用类似,区别在于本地方法栈服务于 Native 方法

  • 可能出现的异常:与 Java 虚拟机栈一样。

Java 堆

  • 唯一目的:存放对象实例

  • 垃圾回收器管理的内存区域

  • 可以处于物理上不连续的内存空间中,但是在逻辑上应该是连续的

  • 可能出现的异常:

    • OutOfMemoryError:Java 堆中没有内存完成实例分配,并且堆也无法再扩展时
  • 虚拟机参数设置:

    • 最大值:-Xmx

    • 最小值:-Xms

    • 两个参数设置成相同值时可避免堆自动扩展

方法区

  • 用于存储已被 Java 虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存等数据

  • 可以选择不实现垃圾收集,换句话说垃圾收集行为在这个区域非常少见,但是对于经常动态性生成大量 Class 的应用,如 Spring 等,需要特别注意类的回收情况

  • 可能出现的异常:

    • OutOfMemoryError:方法区无法满足新的内存分配需求时

运行时常量池

  • 运行时常量池是方法区的一部分
  • Class 文件除了有类的版本、字段、方法、接口等描述信息外,还有一项是常量池表,用于存放编译期生成的各种字面量(就是代码中定义的 static final 常量)和符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中
  • 可能出现的异常:
    • OutOfMemoryError:常量池无法再申请到内存时

直接内存

  • 不属于 Java 虚拟机运行时数据区域的一部分,放到这里讲是因为这部分内存在使用的时候也可能导致 OutOfMemoryError 异常的出现:各个内存区域总和大于物理内存限制(包括物理的和操作系统级的限制),从而导致动态扩展时出现 OutOfMemoryError 异常

  • JDK1.4 的 NIO 类可以使用 Native 函数库直接分配堆外内存,然后通过一个存储在 Java 堆里面的 DirectByteBuffer 对象作为这块内存的引用进行操作,好处是避免了在 Java 堆和 Native 堆中来回复制数据,能在一些场景中提高性能

  • 虚拟机参数设置:-XX:MaxDirectMemorySize

    • 默认等于 Java 堆最大值,即-Xmx指定的值

HotSpot 虚拟机堆中的对象

介绍完 Java 虚拟机的运行时数据区域后,我们大致了解的 Java 虚拟机内存模型的概况。这一小节将介绍 HotSpot 虚拟机在 Java 堆中对象分配、布局和访问的全过程。

对象的创建(new)

当虚拟机遇到 new 指令时:

  1. 检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并检查这个符号引用代表的类是否已经被加载、解析和初始化过。如果没有,则先把这个类加载进内存
  2. 类加载检查通过后,虚拟机将为这个新的对象分配内存,内存的大小在类加载完成后确定
  3. 在 Java 堆中为新对象分配可用内存
  4. 内存分配完成后,虚拟机将分配到的内存空间都初始化为零值
  5. 虚拟机设置对象头中的数据
  6. 此时,从虚拟机的角度看,对象已经创建好了,但从 Java 程序角度看,对象创建才刚刚开始,构造函数还没有执行

第 3 步中,为对象分配可用内存时,会涉及两个问题:

  1. 内存分配方式
     - 指针碰撞(Java 堆中的内存是绝对规整的)
       - 所有被使用过的内存放在一边,没有被使用过的内存放在另一边,中间放一个指针,作为分界点的指示器,那所分配的内存就仅仅是把那个指针向空闲内存空间方向挪一段与对象大小相等的距离
     - 空闲列表(Java 堆中被使用的内存和空闲内存相互交错在一起)
       - Java 虚拟机需要维护一个列表,记录上哪些内存块是可用的,在分配的时候从列表中找一块大小足够大的空间划分给对象实例
  2. 在并发情况下虚拟机创建对象也并不是线程安全的,可能出现正在给对象 A 分配内存,指针还没来得及修改,对象 B 又同时使用了原来的指针来分配内存的情况
     - 对分配内存空间的动作进行同步处理( 采用 CAS 配上失败重试的方式保证更新操作的原子性 )
     - 把内存分配的动作按照线程划分在不同的空间中进行(每个线程在 Java 堆中预先分配一块小内存(这个小内存称为**本地线程分配缓冲区**),哪个线程要分配内存,就在哪个线程的本地线程分配缓冲区中分配,只有这个本地线程分配缓冲区分配完了,分配新的本地线程分配缓冲区才需要同步锁定)
       - 虚拟机参数设置:`-XX:+/-UseTLAB`

对象的内存布局

在 HotSpot 虚拟机中,对象在堆内存中的存储布局分为三部分:

  • 对象头(包括两类信息)
    • 用于存储对象自身的运行时数据,如 HashCode、GC 分代年龄、锁状态标志、线程持有的锁、偏向线程 ID、偏向时间戳等
    • 类型指针,即对象指向它的类型元数据的指针,虚拟机通过这个指针来确定该对象是哪个类的实例。
  • 实例数据(存储我们在程序代码中定义的各种类型的字段内容)
    • 这部分数据受到虚拟机分配策略参数(-XX:FieldsAllocationStyle)和字段在代码中定义顺序的影响
  • 对齐填充(没有实际意义,起到占位符的作用)

对象的访问定位

我们创建对象之后自然是要使用对象,Java 程序会通过栈上的 reference 数据来操作堆上的具体对象(reference 是一个指向对象的引用)。

主流的对象访问方式有两种:

  • 通过句柄访问(reference 中存储的是稳定的句柄地址,在对象被移动的时候只会改变句柄中的实例数据指针,而 reference 本身不需要被修改)
image
  • 通过直接指针访问(速度快得一批,节省了一次指针定位的时间开销)
image

本文小结

本文从概念上介绍了 Java 虚拟机内存的各个区域以及这些区域的作用、服务对象和其中可能出现的异常等,还介绍了虚拟机创建对象(new)的过程、对象的内存布局和如何访问对象。

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

推荐阅读更多精彩内容