#
Java Runtime 数据区
1.程序计数器
defination: 字节码解释器通过改变计数器的值选取下一条需要执行的字节码指令。
每条线程都会有一个独立的程序计数器,互不影响,这部分内存区域又叫“线程私有”
2.Java虚拟机栈
线程私有,生命周期与线程相同
虚拟机栈描述java方法执行的内存模型: 方法执行同时 创建一个Stack Frame 用于存储局部变量表,操作数栈,动态链接,方法出口等信息。
一个方法从调用到执行完成:这个Stack Frame在虚拟机栈入栈到出栈的过程
局部变量表存放编译期可知的各种基本数据类型(boolean,byte,int,short,double,long,float,char),对象引用和returnAddress类型
本区域的异常:线程请求的栈深度大于虚拟机允许时 StackOverflowError,虚拟机扩展时内存不够抛出OutOfMemoryError
3.本地方法栈
类似虚拟机栈,区别是为虚拟机执行本地方法服务。
异常也与java虚拟机栈一致
Sun hotSpot虚拟机直接把本地方法栈和java虚拟机栈合二为一
4.Java堆
所有线程共享的一块内存区域,用于存放对象实例。在虚拟机启动时创建
异常:堆中没有内存完成实例分配,且堆无法再扩展时,抛出OutOfMemoryError
设置:-Xmx -Xms
5.方法区
线程共享,用于存储已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据
原则上,如何实现方法区属于虚拟机实现细节
5.1.运行时常量池
方法区的一部分,Class文件有一项信息是常量池(Constant Pool table)。用于存放编译器生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运行时常量池中存放
运行期间也可能将新的常量放入池中,如String类的intern方法
异常:超过方法区的内存限制抛出OutOfMemoryError
设置 -XX:MaxPermSize
P.S.直接内存
并不是数据区的一部分 但是也可能导致OutOfMemoryError出现。NIO类 引入通道于缓冲区的I/O方式,可以使用Native函数库直接分配对外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象最为这块内存的引用进行操作