理解JVM运行时的数据区是Java编程中的进阶部分。我们在开发中都遇到过一个很头疼的问题就是OutOfMemoryError(内存溢出错误),但是如果我们了解JVM的内部实现和其运行时的数据区的工作机制,那么前面的问题就会迎刃而解。在这片文章中,我们将简单了解JVM中有哪些运行时数据区以及这些数据区的工作机制。
JVM运行时数据区分类
- 程序计数器 (Program Counter (PC) Register)
- JVM栈 (Java Virtual Machine Stacks)
- 堆内存 (Heap Memory)
- 方法区 (Method Area)
- 运行时常量池 (Run-time Constant Pool)
- 本地方法栈 (Native Method Stacks)
有图才能说
按线程持有划分
查看上面的图,可以得知以上六个数据区其实线程私有还是共享,可以分为如下两种。
- 单个线程私有(Managed Per-Thread) 属于这一种的数据区包含 程序计数器, JVM栈还有本地方法栈。 每个线程都私有这三个数据区,这些数据区在其所属的线程创建时初始化,并随着所属线程结束被销毁。
- 多个线程共享 属于这一种的数据区包含 堆内存,方法区和运行时常量池。这些数据区可以被每一个线程访问,他们随着JVM启动而初始化,同时伴随JVM关闭而销毁。
程序计数器
在通用的计算机体系中,程序计数器用来记录当前正在执行的指令,在JVM中也是如此。程序计数器是线程私有,所以当一个新的线程创建时,程序计数器也会创建。由于Java是支持多线程,Java中的程序计数器用来记录当前线程中正在执行的指令。如果当前正在执行的方法是本地方法,那么此刻程序计数器的值为undefined。注意这个区域是唯一一个不抛出OutOfMemoryError的运行时数据区。
JVM栈
在介绍JVM栈之前,简单介绍一个概念,栈帧
栈帧
一个栈帧随着一个方法的调用开始而创建,这个方法调用完成而销毁。栈帧内存放者方法中的局部变量,操作数栈等数据。
JVM栈只对栈帧进行存储,压栈和出栈操作。栈内存的大小可以有两种设置,固定值和根据线程需要动态增长。在JVM栈这个数据区可能会发生抛出两种错误。
- StackOverflowError 出现在栈内存设置成固定值的时候,当程序执行需要的栈内存超过设定的固定值会抛出这个错误。
- OutOfMemoryError 出现在栈内存设置成动态增长的时候,当JVM尝试申请的内存大小超过了其可用内存时会抛出这个错误。
堆数据区
堆数据区是用来存放对象和数组(特殊的对象)。堆内存由多个线程共享。堆内存随着JVM启动而创建。众所周知,Java中有一个很好的特性就是自动垃圾回收。垃圾回收就操作这个数据区来回收对象进而释放内存。如果堆内存剩余的内存不足以满足于对象创建,JVM会抛出OutOfMemoryError错误。
方法区
在JVM规范中,方法区被视为堆内存的一个逻辑部分。这一点可能由于具体的JVM实现而不同,甚至在方法区不实现垃圾回收处理也是可以的。方法区和堆内存一样被多个线程访问,方法区中存放类的信息,比如类加载器引用,属性,方法代码和构造方法和常量等。当方法区的可用内存无法满足内存分配需求时,JVM会抛出OutOfMemoryError错误。
运行时常量池
运行时常量池创建在方法区,当一个类或者一个接口被创建的时候,JVM会创建一个运行时常量池。一个运行时常量池实际上是一个类或者接口的class文件中常量池表(constant_pool table)的运行时展示形式。一个运行时常量池包含了多种类型的常量,从诸如运行时可以确定的数值型字面量到运行时才能决定的方法和属性引用。当运行时常量池无法满足于内存分配需求时,JVM会抛出OutOfMemoryError错误。
本地方法栈
一个支持native方法调用的JVM实现,需要有这样一个数据区,就是本地方法栈,Java官方对于本地方法的定义为methods written in a language other than the Java programming language
,就是使用非Java语言实现的方法,但是通常我们指的一般为C或者C++,因此这个栈也有着C栈这一称号。一个不支持本地方法执行的JVM没有必要实现这个数据区域。本地方法栈基本和JVM栈一样,其大小也是可以设置为固定值或者动态增加,因此也会对应抛出StackOverflowError和OutOfMemoryError错误。
相关推荐
第3节: 揭秘JVM运行时数据区-02第3节: 揭秘JVM运行时数据区-02第3节: 揭秘JVM运行时数据区-02第3节: 揭秘JVM运行时数据区-02第3节: 揭秘JVM运行时数据区-02第3节: 揭秘JVM运行时数据区-02第3节: 揭秘JVM运行...
### JVM运行时数据区详解 #### 一、概述 Java虚拟机(JVM)作为Java程序的运行环境,定义了一系列用于程序执行过程中使用的数据区域。这些数据区域在JVM启动时创建,在JVM退出时销毁。其中一些数据区域是线程独立...
Java内存模型与JVM运行时数据区的区别详解 Java内存模型是Java语言在多线程并发情况下对于共享变量读写的规范,主要是为了解决多线程可见性、原子性的问题,解决共享变量的多线程操作冲突问题。Java内存模型是为了...
JVM运行时数据区原理解析 JVM运行时数据区是Java虚拟机中最重要的组成部分之一,它是JVM运行时的核心区域,负责存储和管理程序运行时需要的数据。根据JVM规范,JVM运行时数据区可以分为五个部分:PC寄存器、虚拟机...
"JVM运行时数据区划分原理详解" JVM运行时数据区划分原理详解是Java虚拟机(JVM)的核心组件之一,负责管理Java应用程序的内存资源。该原理详解了JVM如何划分和管理内存空间,以便更好地支持Java应用程序的运行。 ...
首先,我们来看JVM运行时数据区的组成部分: 1. **方法区(Method Area)**:这是一个全局共享的内存区域,存储了类信息、常量、静态变量以及编译后的代码。这部分内存被称为Non-Heap,当其内存不足时,会抛出`...
此外,直接内存不属于JVM运行时数据区的一部分,但被频繁使用,如在Java NIO中用于基于Channel与Buffer的IO操作。 3. JVM执行引擎与垃圾回收: JVM的执行引擎负责解释字节码指令,它通常包括即时编译器和解释器两种...
03 JVM 运行时数据区概述及线程的 PPT 重绘。讲述 Java 虚拟机 运行时数据区所处位置,结构划分,以及各个区域与线程的关系。
### JVM运行时数据区 #### 程序计数器 程序计数器是当前线程所执行的字节码的行号指示器,是线程私有的内存区域。由于Java是多线程并发执行的,每个线程都需要有一个独立的程序计数器,以便记录线程切换后继续执行...
5. **JVM运行时数据区详解**: - 程序计数器:每个线程都有自己的计数器,用于记录当前线程正在执行的字节码指令位置。 - 虚拟机栈:存储每个方法的局部变量表、操作数栈、动态链接和方法出口等信息,每个方法调用...
直接内存并不属于JVM运行时数据区的一部分,而是由JDK NIO库直接分配的堆外内存。这部分内存不受JVM堆大小的限制,但同样可能导致内存溢出。DirectByteBuffer对象作为直接内存的引用,用于操作这块内存。 #### 三、...
#### 三、JVM运行时数据区详解 ##### 3.1 PC寄存器 PC寄存器(Program Counter Register)用于存储当前线程下一条指令的位置。对于执行native方法的线程,PC寄存器内容为空。 ##### 3.2 JVM栈 JVM栈是线程私有的...
#### 二、JVM运行时数据区 JVM在运行时会划分多个内存区域来管理数据,主要包括以下几个部分: - **方法区**:存储类的信息(包括类的方法和字段)、常量、静态变量等。 - **堆区**:用于存放所有类实例和数组对象...
6. 直接内存(Direct Memory):并非JVM运行时数据区的一部分,它通过NIO类库支持直接操作本地内存,并不是JVM规范所定义的。 JVM内存模型的设计是为了支持Java程序的动态特性,同时保证内存的安全和高效利用。JVM...
以下是对JVM运行时数据区及其相关知识点的详细说明: 1. **运行时数据区**: - **程序计数器**:每个线程都有自己的程序计数器,用于存储当前线程执行的字节码的行号,是唯一没有规定内存溢出情况的区域。 - **...
5. **JVM运行时数据区详解**: - 程序计数器:记录当前线程执行的字节码指令位置,用于控制程序流程。 - 虚拟机栈:每个方法对应一个栈帧,存储局部变量表、操作数栈、动态链接和方法返回地址等信息。 - 本地方法...
#### JVM运行时数据区 **5. JVM运行时数据区详解** JVM管理的内存主要分为以下几部分: - **程序计数器(Program Counter Register)**:记录当前线程所执行的字节码指令的位置,确保线程恢复执行时可以从正确的...
#### 二、JVM运行时数据区 ##### 2.1 堆区 堆是所有线程共享的一块内存区域,在虚拟机启动时创建。主要用于存放对象实例和数组。 - **新生代**:主要存放新创建的对象。 - **老年代**:经过多次GC后存活的对象会被...