1.运行时数据区域
线程共享:堆,方法区(包括运行时常量池), 直接内存
线程私有:虚拟机栈,本地方法栈,程序计数器。
程序计数器:
唯一一个不会出现OutOfMemoryError的内存区域,生命周期和线程相同。
1.字节码解释器通过改变程序计数器来依次读取指令,从而实现代码的流程控制。
2.在多线程的情况下,程序计数器用于记录当前线程执行的位置,从而线程切换后能恢复到正确的执行位置。
虚拟机栈
生命周期和线程相同,描述方法执行的内存模型。
本地方法栈
为虚拟机使用到的 Native 方法服务。
堆
Java 虚拟机所管理的内存中最大的一块,Java 堆是所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例以及数组都在这里分配内存。
Java 堆是垃圾收集器管理的主要区域,因此也被称作GC堆(Garbage Collected Heap).从垃圾回收的角度,由于现在收集器基本都采用分代垃圾收集算法,所以Java堆还可以细分为:新生代和老年代。
在 JDK 1.8中移除整个永久代,取而代之的是一个叫元空间(Metaspace)的区域(永久代使用的是JVM的堆内存空间,而元空间使用的是物理内存,直接受到本机的物理内存限制)。
方法区
它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是它却有一个别名叫做 Non-Heap(非堆),目的应该是与 Java 堆区分开来。
运行时常量池
运行时常量池是方法区的一部分。Class 文件中除了有类的版本、字段、方法、接口等描述信息外,还有常量池信息(用于存放编译期生成的各种字面量和符号引用)
既然运行时常量池时方法区的一部分,自然受到方法区内存的限制,当常量池无法再申请到内存时会抛出 OutOfMemoryError 异常。
JDK1.7及之后版本的 JVM 已经将运行时常量池从方法区中移了出来,在 Java 堆(Heap)中开辟了一块区域存放运行时常量池
直接内存
直接内存并不是虚拟机运行时数据区的一部分,也不是虚拟机规范中定义的内存区域,但是这部分内存也被频繁地使用。而且也可能导致 OutOfMemoryError 异常出现。
JDK1.4中新加入的 NIO(New Input/Output) 类,引入了一种基于通道(Channel) 与缓存区(Buffer) 的 I/O 方式,它可以直接使用Native函数库直接分配堆外内存,然后通过一个存储在 Java 堆中的 DirectByteBuffer 对象作为这块内存的引用进行操作。这样就能在一些场景中显著提高性能,因为避免了在 Java 堆和 Native 堆之间来回复制数据。
本机直接内存的分配不会收到 Java 堆的限制,但是,既然是内存就会受到本机总内存大小以及处理器寻址空间的限制。
参考:JavaGuide
本文详细介绍了Java虚拟机的运行时数据区域,包括线程共享的堆、方法区(含运行时常量池)、直接内存,以及线程私有的虚拟机栈、本地方法栈和程序计数器。讨论了各个区域的作用,如程序计数器用于记录线程执行位置,堆是对象实例的主要存储区域,方法区存储类信息等。此外,还提到了直接内存的使用和垃圾收集对Java堆的影响,以及JDK1.8中方法区的变化和运行时常量池的迁移。
2880

被折叠的 条评论
为什么被折叠?



