JVM中运行时数据区域(内存结构)

本文介绍了Java虚拟机的运行时数据区域,包括程序计数器、虚拟机栈、本地方法栈、Java堆、方法区、运行时常量池和直接内存。这些区域各自承担不同的功能,如线程私有区域的程序计数器和虚拟机栈,以及线程共享的Java堆和方法区。堆内存主要用于对象实例的分配,而方法区存储类信息、常量等。直接内存允许通过通道和缓冲区直接分配堆外内存,提高效率。
摘要由CSDN通过智能技术生成

Java虚拟机所管理的内存通常包括以下几个运行时数据区域,如图所示:
Java虚拟机运行时数据区

1. 程序计数器

程序计数器是一块较小的内存空间,可看做是当前线程所执行的字节码的行号指示器。在虚拟机的概念模型里,字节码解释器工作时就是通过改变这个计数器的值来选择下一条需要执行的字节码指令。分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖这个计数器来完成。

因为Java里面是支持多线程的,所以要求每一个线程都有一个自己的程序计数器,以便线程切换后能恢复到正确的执行位置。各线程之间互不影响,独立存储,我们称这类内存区域为“线程私有”的内存。

此内存区域没有规定任何OutOfMemoryError情况的区域。

2. 虚拟机栈

虚拟机栈也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的同时会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。我们只需要知道每个线程都会有一个虚拟机栈,然后线程里的多个方法就对应栈里面的栈帧(栈存储元素的基本单位),每个方法从调用到执行完成对应一个栈帧在虚拟机栈中入栈到出栈的过程。

通常有人把Java内存分为堆内存和栈内存,其实这两块内存只是与一些对象的内存分配有关,比较粗糙。比如通常Java程序中的所有对象均是在堆中分配的,它是一个线程共享的区域,如上图所示;而栈则是我们所说的虚拟机栈,通常用来供方法内的局部变量进行分配,也就是上面提到的栈帧中的局部变量表。

局部变量表存放了编译器可知的各种基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用(reference类型,它不等同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或其他与此对象相关的位置)和returnAddress类型(指向了一条字节码指令的地址)。

Slot(局部变量空间)是局部变量表的基本单位(通常为32位),64位的long和double类型数据会占用2个局部变量空间,其余的数据类型只占用1个。局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法需要在栈帧中分配多大的局部变量空间是完全确定的,在方法运行期间不会改变局部变量表的大小。

此内存区域规定了两种异常状况:若线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常(一个方法可能递归请求了大量的方法且还未返回,使得请求的方法个数大于栈的深度,导致栈内放不了那么多的栈帧);如虚拟机可以动态扩展,且扩展时无法申请到足够内存,则会抛出OutOfMemoryError异常。

3. 本地方法栈

与虚拟机栈其实是类似的,只是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则为虚拟机使用到的Native方法服务。与虚拟机栈一样,本地方法栈区域也会抛出StackOverflowError和OutOfMemoryError异常。

4. Java堆

Java 堆算是Java虚拟机所管理的内存中最大的一块。Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建,此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。Java虚拟机规范中描述的是:所有的对象实例以及数组都要在堆上分配,当然随着技术发展,这个也就不是那么绝对了。

Java堆是垃圾收集器管理的主要区域,因此很多时候也被称为“GC堆”。因为很多对象都是在这里存储的,也就需要在这里进行回收。从内存回收的角度看,由于现在收集器基本都采用分代收集算法,所以Java堆中还可以细分为:新生代老年代,这个在后面文章中会讲到,再详细一些还有Eden空间、From Survivor空间、To Survivor空间等。从内存分配的角度看,线程共享的Java堆中可能划分出多个线程私有分配缓冲区(TLAB)。不过无论如何划分,都与存放内容无关,无论哪个区域,存储的都仍然是对象实例,进一步划分的目的是为了更好地回收内存,或者更快地分配内存。

Java虚拟机规范规定,Java堆可以处于物理上不连续的内存空间中,只要逻辑上是连续的即可。

5. 方法区

方法区也是线程共享的,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。要注意“方法区”和“永久代”并不是一个东西,仅仅是因为HotSpot虚拟机的设计团队选择把GC分代收集扩展至方法区,或者说使用永久代来实现方法区而已,这样HotSpot的垃圾收集器就可以像管理Java堆一样管理这部分内存

其实垃圾收集行为在这个区域是比较少出现的,这区域的内存回收目标主要是针对常量池的回收对类型的卸载。而且回收成绩难以让人满意,尤其是类型的卸载条件相当苛刻,但这部分区域的回收确实有必要。此区域有可能会抛出OutOfMemoryError异常。java对象的“死活”中描述了一些关于方法区回收的细节。

6. 运行时常量池

运行时常量池是方法区的一部分。Class文件中会有一项信息是常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中存放

Java虚拟机规范没有对运行时常量池做任何细节的要求,但一般来说,除了保存Class文件中描述的符号引用外,还会把翻译出来的直接引用也存储在运行时常量池中

运行时常量池相对于Class文件常量池的另外一个重要特征就是具备动态性,Java语言并不要求常量一定只有编译器才能产生,也就是并非预置入Class文件中常量池的内容才能进入方法区运行时常量池,运行期间也可将新的常量池放入池中,这种特性被开发人员利用的比较多的是String类的intern()方法

7. 直接内存

直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域。但是这部分内存也会被频繁的使用。在JDK1.4中新加入了NIO(New Input/Output)类,引入了一种基于通道缓冲区的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作,这样避免了在Java堆和Native堆中来回复制数据

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值