JVM——Java内存区域与溢出异常
JVM内存区域的划分?
一、内存区域划分
线程私有:程序计数器,本地方法栈,虚拟机栈
线程共享:Java堆,方法区,运行时常量池
1、程序计数器:当前线程所执行的字节码的行号指示器(执行Java方法,记录正在执行的虚拟机字节码指令地址;执行Native方法,计数器值为空)
程序计数器是JVM中唯一一块不会产生OOM异常的区域
2、虚拟机栈(方法栈):Java方法执行的内存模型。每个方法执行的同时会创建一个栈帧用于存储局部变量表
,操作数栈、方法出口等信息。(每个方法从开始执行到结束的过程,对应虚拟机栈入栈出栈的过程)
局部变量表:8大基本数据类型+对象引用,编译期间完成内存分配,运行期间所占空间确定,不会改变局部变量表的大小。
异常
:
①线程请求栈深度大于JVM允许的深度:StackOverFlowError
②JVM进行栈的动态扩展,无法申请到足够内存:OOM
3、本地方法栈:本地(native)方法执行的内存模型
4、Java堆:JVM启动时创建,所有线程共享此内存,可以处于物理上不连续的空间,是垃圾回收器管理的最主要区域。
所有的对象实例以及数组都分配在堆上
异常
:若堆上没有足够的内存完成对象实例分配并且堆无法再扩展时,抛出OOM(堆溢出)
5、方法区(静态区、全局区):用于存储已被虚拟机加载的类信息(类中的方法等)、常量、静态变量等。
异常
:方法区无法满足内存分配要求时,抛出OOM
6、运行时常量池:方法区的一部分,存放字面量与符号引用
字面量:字符串常量(JDK1.7后移到堆中),final常量、基本数据类型的值
符号引用:类的完全限定名、字段/方法的名称与描述符
异常
:异常情况与方法区相同,本身就是方法区的一部分
内存泄漏与内存溢出的区别?