JVM运行时数据区域

程序计数器

1、 线程所执行的字节码的行号指示器;
2、JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式;
3、 每条线程都需要有一个独立的程序计数器,所以是线程私有的内存区域;
4、 执行Java方法,记录的是正在执行的虚拟机字节码指令的地址;
5、 执行Native方法,计数器值为空;
6、 唯一一个在JVM规范中没有规定任何OOM情况的区域;

Java虚拟机栈

1、 是线程私有的,其生命周期与线程相同;
2、 是Java方法执行的内存模型;
3、 每个方法执行的同时会创建一个栈帧,每个方法从调用到执行完成的过程,对应着一个栈帧在虚拟机栈中从入栈到出栈;
4、 栈帧存放内容:局部变量表、操作数栈、动态链接、方法出口;
5、 局部变量表存放内容:编译期可知的基本数据类型(byte,short,int,long,float,double,boolean,char)、对象引用、returnAddress类型;
6、 64位长度的long和double类型的数据会占用2个局部变量空间,其余的占用1个;
7、 局部变量表所需内存空间大小,在编译期完成分配,方法运行期间不会改变;
8、 Java虚拟机栈会存在以下两种类型的异常:

异常类型 原因
StackOverflowError 线程所申请的栈深度大于虚拟机所允许的最大深度
OutOfMemoryError JVM栈可以动态扩展,如果扩展时无法申请到足够的内存

本地方法栈

1、 作用与Java虚拟机栈相似,为JVM使用的Native方法服务;
2、 有的虚拟机(例如Sun的HotSpot)把本地方法栈和虚拟机栈合二为一;
3、 也会抛出两种类型的异常:StackOverflowError和OutOfMemoryError;

Java堆区

1、 JVM管理的内存中最大的一块;
2、 是被所有线程共享的一块内存区域;
3、 虚拟机启动时创建;
4、 唯一目的:存放对象实例(所有的对象实例和数组都是在堆上分配);
5、 是垃圾收集器管理的主要区域,也被称为“GC堆”;
6、 采用分代收集算法,Java堆细分为:新生代、老年代。再细致一点的有:Eden空间、From Survivor空间、To Survivor空间;
7、 Java堆可以处于物理上不连续的内存空间中,只要逻辑连续即可;
8、 通过JVM参数 -Xmx和-Xms控制扩展;
9、 异常类型

异常类型 原因
OutOfMemoryError 堆中没有内存完成实例分配,且堆也无法再扩展

方法区

1、 与堆一样,也是各个线程共享的内存区域;
2、 存储已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码;
3、 HotSpot中称为永久代;
4、 JDK1.7中已经把原本在永久代中的字符串常量池移出;
5、 该区域内存回收的目标主要是:针对常量池的回收和对类型的卸载;
6、异常类型

异常类型 原因
OutOfMemoryError 方法区无法满足内存分配需求时
原文地址:https://www.cnblogs.com/muscleape/p/10229226.html