一、内存模型

一、内存模型

程序计数器、虚拟机栈、本地方法栈、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来优化

整体预览图:


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容