第二章:Java内存区域与内存溢出异常
Java虚拟机管理的内存包含以下运行时数据区域:
- 方法区(Method Area)
- 虚拟机栈(VM Stack)
- 本地方法栈(Native Method Stack)
- 堆(Heap)
- 程序计数器(Program counter Register)
程序计数器
程序计数器是当前线程所执行的字节码的行号指示器。字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。分支,循环,异常,线程恢复都需要以来这个计数器。
每个线程都需要一个独立的程序计数器。
Java虚拟机栈
Java虚拟机栈的生命周期和线程相同。每个方法在执行的时候会创建一个栈帧用于存储局部变量表,操作数栈,动态链接,方法出口等。
局部变量表存放了各种基本数据类型,对象引用指针。局部变量表所需的内存空间在编译器完成分配,进入方法时,所需要分配多大的局部变量空间时完全确定的。
如果线程请求的栈深度大于虚拟机所需要的深度,抛出StackOverFlowError
异常。
如果虚拟机栈动态扩展时无法申请到足够的内存,会抛出OutOfmemoryError
异常。
本地方法栈
本地方法栈和虚拟机栈的作用类似,一个是执行本地方法的,一个是执行java方法的。
Java堆
Java堆是Java虚拟机中所管理的内存最大的一块,Java堆是被所有线程共享的一块内存区域。唯一目的就是存放对象实例。
Java堆是垃圾收集器管理的主要区域。由于现在收集器基本采用分代收集算法,所以Java堆分为新生代,老年代。
如果堆中没有内存完成实例分配,会抛出OutOfmemoryError
异常。
方法区
方法区也是线程共享的内存区域。用于存储虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码。方法区被描述为堆的一个逻辑部分。
运行时常量池
运行时常量池是方法区的一部分。用于存放编译期生成的各种字面量和符号引用,再类加载后进入方法去的运行时常量池存放。