一、内存模型
程序计数器、虚拟机栈、本地方法栈、java堆、方法区
1、程序计数器
属于线程的私有内存空间,伴随线程创建。每个线程都有一个独立的程序计数器,用于记录下一条要运行的指令,各线程间互相独立。如果当前线程正在执行一个java方法,则程序计数器记录正在执行额java字节码地址,如果当前线程正在执行一个native方法,则程序计数器置为空
2、java虚拟机栈
属于线程的私有内存空间,伴随线程创建,保存了方法的局部变量、部分结果、并参与方法的调用和返回。java栈大小可以是动态或固定的。会引起 StackOverFlowError和OutOfMemeryError。
2.1如果线程请求的栈深度超过最大可用的栈深度,会抛出StackOverFlowError;如果栈可以动态扩展,而在扩展过程中,没有足够的内存空间来支持栈的空间,则会抛出OutOfMemeryError。
比如死循环递归:
int i =0;
static void concreate()
{
i++;
concreate(); //会由于栈调用深度导致栈溢出
}
main (){
concreate();
}
当设置 -Xss后,可以增加部分调用深度。
虚拟机栈在运行时采用 栈帧的数据结构来存储上下文数据,当调用深度很深时,会导致内存膨胀。如图:
3、本地方法栈
本地方法栈类似于虚拟机栈,也属于线程私有内存空间,java虚拟机栈用于管理java函数调用,而本地方法栈用于管理本地方法的调用,本地方法不是java实现,而是C实现。也会抛出堆栈异常和内存异常。
4、java堆
属于所有线程共享的部分,几乎所有的对象和数组都是在堆中分配空间的,java堆分为新生代和老年代,新生代用于存放刚刚产生的和年轻的对象,如果对象一直未回收,就会被移到老年代。新生代又分为Eden、s0、s1,s0和s1属于survivor区,也就是其中存储的对象至少经过了一次垃圾回收。可以通过-Xmx和-Xms控制,属于GC的重点范围
5、方法区
与堆空间类似,属于所有线程共享的部分,保存的信息是类的元数据。方法区中最为重要的是类的类型信息、常量池、域信息、方法信息。类型信息包括类的完整名称、父类的完整名称、类型修饰符、类的直接接口类表;常量池包括类方法、域等信息所引用的常量信息;域信息包括域名称、域类型、域修饰符;方法信息包括方法名称、返回类型、方法参数、方法修饰符、方法字节码 ,保存的信息大部分来源于.class文件,在hot spot 虚拟机中,也称谓永久区,对永久区的回收通常为:1.对永久区常量池的回收 2.永久区对类元数据的回收
设置 PermSize 和MaxPermSize来优化
整体预览图: