JVM运行时数据区域
1 程序计数器
1.1 概念
一块较小的内存空间,当前线程所执行的字节码的行号指示器。
1.2 用途
应用在字节码解释器中,以便获取到下一条字节码指令。
字节码解释器如何工作:改变程序计数器的值,从而选取到下一条需要执行的字节码指令
1.3 特点
每条线程都有一个独立的程序计数器,各线程之间计数器互不影响,独立存储(线程私有的内存)
Java虚拟机多线程本质:线程轮流切换并分配处理器执行时间
一个确定时刻,一个处理器(一个内核),执行一个线程中的指令
1.4 存储内容
Java方法:记录正在执行的虚拟机字节码指令的地址
Native方法:计数器值为空(Undefined)
1.5 异常
唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域
2 Java虚拟机栈
2.1 概念与特点
与程序计数器相同,线程私有,生命周期与线程相同
2.2 用途
是Java方法执行的内存模型
java方法内存模型:
方法执行同时创建一个栈帧
方法从调用到执行完成==》一个栈帧入栈到出栈(栈是指虚拟机栈)
栈帧:
存储局部变量表、操作数栈、动态链接、方法出口等信息
2.3 存储
“栈”=局部变量表
存放类型:
- 基本数据类型
- 对象引用
- returnAddress类型
大小:
64位长度的long和double类型——>2个局部变量空间(Slot)
其余数据类型——>1个局部变量空间(Slot)
分配:
编译期间完成分配,在栈中的局部变量空间是完全确定的
方法运行期间不会改变局部变量表的大小
2.4 异常
- 线程请求深度大于虚拟机所允许的深度,——>StackOverflowError异常
- 虚拟机栈可以动态扩展时,扩展时无法申请到足够的内存,——>OutOfMemoryError异常
3 本地方法栈
3.1 概念
类似于虚拟机栈,本地方法栈为Native方法服务
虚拟机栈为Java方法服务
3.2 异常
与虚拟机栈相同:StackOverflowError异常和OutOfMemoryError异常
4 Java堆
4.1 概念
内存中最大的一块,所有线程共享,虚拟机启动时创建
4.2 存储
所有的对象实例以及数组
特点:
- 可处于物理上不连续的内存空间,逻辑上连续即可
- 可选择固定大小或可扩展
当前主流的虚拟机都是按照可扩展来实现的(通过**-Xmx和-Xms**控制)
4.3 分类
Java堆也被称为“GC堆”
内存回收角度:
- 新生代
- 老年代
内存分配角度:
多个线程私有的分配缓冲区
4.4 异常
堆中没有内存完成实例分配,并且堆也无法扩展——>OutOfMemoryError异常
5 方法区
5.1 概念
与Java堆一样,各线程共享的内存区域
5.2 存储
存储已被虚拟机加载的类信息、常量、静态常量、即时编译器编译后的代码等数据
特点:
- 不需要连续的内存
- 可选择固定大小或可扩展
- 可选择不实现垃圾收集
5.3 异常
方法区无法满足内存分配需求——>OutOfMemoryError异常
6 运行时常量池
6.1 概念
方法区的一部分
6.2 存储
编译期生成的各种字面量和符号引用
Class文件:类的版本、字段、方法、接口等描述性信息,还有一项信息是常量池
编译期生成的各种字面量和符号引用,原本在常量池中存放,类加载后这部分内容才进入方法区的运行时常量池中存放
6.3 Class文件常量池与运行时常量池区别
Class文件常量池 | 运行时常量池 |
---|---|
格式有严格规定 | 没有任何细节要求,可以按照自己的需求来实现它 |
不具备动态性 | 具备动态性,运行期间也能将新的常量放入池中 |
动态性:例如String类的intern()方法
6.4 异常
常量池无法再申请到内存——>OutOfMemoryError异常
7 直接内存
7.1 概念
不属于虚拟机运行时数据区
不属于Java虚拟机规范中定义的内存区域
7.2 NIO(New Input/Output)类
功能:使用Native函数库直接分配堆外内存,通过DirectByteBuffer对象进行操作
Java堆中的DirectByteBuffer对象=此内存引用
优点:避免了在Java堆和Native堆中来回复制数据
7.3 存储
只受本机总内存大小和处理器寻址空间的限制
7.4 异常
忽略直接内存——>各个内存区域总和大于物理内存限制——>动态扩展出现OutOfMemoryError异常