【Jvm】运行时数据区域

内容来自《深入理解Java虚拟机》和网络。

本篇结构:

  • 程序计数器

  • Java 虚拟机栈

  • 本地方法栈

  • Java 堆(Java Heap)

  • 方法区

  • 运行时常量池

  • 直接内存

image

一、程序计数器

程序计数器(Program Counter Register) 是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器

在虚拟机的概念模型里,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。

每条线程都需要有一个独立的的程序计数器,各条线程间计数器互不影响,独立存储。我们称这类内存区域为”线程私有”。

在 JVM 规范中规定,如果线程执行的是一个 Java 方法,则程序计数器中保存的是当前需要执行的指令的字节码地址;如果线程执行的是 native 方法,则程序计数器中的值是空(undefined)。

此内存区域是唯一一个在Java 虚拟机中没有规定任何内存泄露(OutOfMemoryError)情况的区域,程序计数器中存储的数据所占空间的大小不会随程序的执行而发生改变。

二、Java 虚拟机栈

与程序计数器一样,Java 虚拟机栈也是线程私有的。

虚拟机栈描述的是Java 方法执行的内存模型:

每个方法在执行的同时会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机中入栈到出栈的过程。

程序每执行一个方法,就会分配一个栈帧,并将建立的栈帧压栈。当方法执行完毕之后,便会将栈帧出栈。因此可知,线程当前执行的方法所对应的栈帧必定位于Java 栈的顶部。这也是为什么在使用递归方法的时候容易导致栈内存溢出的现象了。

[图片上传中...(image-23842f-1560258206098-0)]

在虚拟机规范中,对这个区域规定了两种异常情况:

  • 如果线程请求的栈深度大于虚拟所允许的深度,将抛出 StackOverflowError;

  • 如果虚拟机栈可以动态扩展,但是扩展时无法申请到足够的内存,就会跑出 OutOfMemoryError。

2.1、局部变量表

用来存储在编译器可知的基本数据类型的变量(8种基本数据类型)、对象引用(reference 类型)。

对于引用类型的变量,存的是指向对象起始地址的引用指针和 returnAddress 类型(方法返回地址,当一个方法执行完毕之后,要返回之前调用它的地方,因此在栈帧中必须保存一个方法返回地址)。

局部变量表的大小在编译器就可以确定其大小了,因此在程序执行期间局部变量表的大小是不会改变的。

2.2、操作数栈

一个线程执行方法的过程中,实际上就是不断执行语句的过程,而归根到底就是进行计算的过程,因此可以说,程序中的所有计算过程都是在借助于操作数栈来完成的。

三、本地方法栈

本地方法栈和虚拟机方法栈发挥的作用是非常的相似的,他们之间的区别不过是虚拟机栈为虚拟机执行 Java 方法而服务的,本地方法栈是为虚拟机执行 Native 方法而服务的。

在JVM规范中,并没有对本地方发展的具体实现方法以及数据结构作强制规定,有的虚拟机(譬如:Sun HotSpot虚拟机)直接将虚拟机栈和本地方法栈合二为一。

与虚拟机栈一样,本地方法栈也会抛出 StackOverflowError 异常和OutOfMemoryError 异常。

四、Java 堆(Java Heap)

对于大多数应用来说,Java 堆是虚拟机所管理的内存中最大的一块,Java 堆是所有线程所共享的内存区域;在Java虚拟机启动时创建。Java 堆的唯一目的就是存放对象实例,几乎所有的对象实例以及数组都是在这里分配内存。

此内存区域 的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存,在一点在 Java 虚拟规范中描述:所有的对象以及数组实例都要在堆上分配内存,但随着 JIT 编译技术的发展以及逃逸技术逐步成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化,所有的对象都分配在堆上也渐渐的变得不是那么绝对。Java是垃圾收集器管理的主要区域,因此很多时候被称为 “GC” 堆。

从内存回收的角度看,java 堆可分为:新生代(Eden Space、From Survivor Space、To Survivor Space)和老年代。

如果在堆中没有内存完成实例分配并且堆也无法再扩展时将会抛出 OutOfMemoryError。

五、方法区

方法区在 JVM 中也是一个非常重要的区域,它与堆一样,是被线程共享的区域。

在方法区中,存储了每个类的信息(包括类的名称、方法信息、字段信息)、静态变量、常量以及编译器编译后的代码等。

方法区(method area)是 JVM 规范中定义的一个概念,用于存储类信息、常量池、静态变量、JIT 编译后的代码等数据,具体放在哪里,不同的实现可以放在不同的地方。而永久代Hotspot 虚拟机特有的概念(HotSpot 虚拟机以永久代来实现方法区,从而 JVM 的垃圾收集器可以像管理堆区一样管理这部分区域,从而不需要专门为这部分设计垃圾回收机制),是方法区的一种实现,别的 JVM 都没有这个东西。Hotspot 1.7 后也去除了持久代。

JVM 规范规定,当方法无法满足内存分配的需要时将会抛出 OutOfMemoryError。

5.1、永久代和方法区的关系

涉及到内存模型时,往往会提到永久代,那么它和方法区又是什么关系呢?

《Java虚拟机规范》只是规定了有方法区这么个概念和它的作用,并没有规定如何去实现它。那么,在不同的 JVM 上方法区的实现肯定是不同的了。 同时大多数用的 JVM 都是 Sun 公司的 Hotspot。在 Hotspot 上把 GC 分代收集扩展至方法区,或者说使用永久代来实现方法区。

因此,可以得出结论,永久代是 Hotspot 的概念,方法区是 Java 虚拟机规范中的定义,是一种规范,而永久代是一种实现,一个是标准一个是实现。其他的虚拟机实现并没有永久带这一说法。在 1.7 之前在(JDK1.2 ~ JDK6)的实现中,HotSpot 使用永久代实现方法区,HotSpot 使用 GC分代来实现方法区内存回收,可以使用如下参数来调节方法区的大小:

-XX:PermSize
方法区初始大小
-XX:MaxPermSize
方法区最大大小
超过这个值将会抛出OutOfMemoryError异常:java.lang.OutOfMemoryError: PermGen

5.2、元空间

对于 Java8, Hotspot 取消了永久代,那么是不是也就没有方法区了呢?当然不是,方法区是一个规范,规范没变,它就一直在。

取代永久代的就是元空间。它和永久代有什么不同的?

  • 存储位置不同,永久代物理是是堆的一部分,和新生代,老年代地址是连续的,而元空间属于本地内存;
  • 存储内容不同,元空间存储类的元信息,静态变量和常量池等并入堆中。相当于永久代的数据被分到了堆和元空间中。

六、运行时常量池

运行时常量池是方法区的一部分。

Class文件除了有类的版本,字段,方法,接口等描述信息之外,还有一项信息是常量池,用于存放编译期生成的字面量和符号引用,这部分内容将在类加载之后进入方法区中的运行时常量池中存放。

它是每一个类或接口的常量池的运行时表示形式,在类和接口被加载到 JVM 后,对应的运行时常量池就被创建出来。Java 语言并不要求常量只有编译时才能产生,也就是并非预置到 class 文件中常量池的内容才能进入到方法区运行时常量池,运行区间也可以把新的常量放到池中,使用得比较多的是 String 的 intern()方法。

当常量池无法申请到内存时将会抛出 OutOfMemoryError。

七、直接内存

直接内存(Direct Memory)不是Java 虚拟机规范中定义的内存区域。

在 JDK1.4 中新加入的 NIO 类,引入了一种基于通道(Channel)和缓冲区(Buffer)的 I/O 形式,他可以使用 Native 函数直接分配堆外内存,然后通过一个存储在 Java 堆中的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样能在一些场所显著提高性能,因为避免了在Java 堆和Native 堆中来回复制数据。

本机直接内存受本机总内存限制。可能导致 OutOfMemoryError。

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

推荐阅读更多精彩内容

  • JVM运行时数据区域 原文:Run-Time Data Areas Java虚拟机定义了在程序执行期间使用的多个运...
    叠最厚的甲阅读 314评论 0 0
  • 《深入理解Java虚拟机》day1 1.程序计数器: 1)他在内存中所占的空间较小,它可以被看作为当前线程所执行的...
    是一动不动的friend阅读 298评论 0 0
  • 运行时数据区 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域都有各...
    lijiaccy阅读 160评论 0 0
  • 此处的内容是根据Java虚拟机规范(Java SE 7)相关内容以及深入理解Java虚拟机等做的总结。可能有不对的...
    加大装益达阅读 286评论 0 2
  • 连续几天痴迷DISC力量,并写出很多的文字,并试图和身边的人深度交流传播,有很多的人鼓励并支持我,当然也有很多人不...
    米肖阅读 734评论 0 51