您的位置:首页 > 编程语言 > Java开发

JVM 内存模型简单分析

2017-01-14 12:11 204 查看

背景

说是简单分析,不如说是一个笔记,写这篇文章完全是处于自己脑子不好使,总是记不住JVM的内存模型,对他的概念一直很模糊,今天终于忍不住,觉得写一篇文章来彻底记住他。(部分来自网上其他作者,我做了个汇总,在此表示感谢!)

模型图



简单分析

程序计数器

多线程时,当线程数超过CPU数量或CPU内核数量,线程之间就要根据时间片轮询抢夺CPU时间资源。因此每个线程有要有一个独立的程序计数器,记录下一条要运行的指令。线程私有的内存区域。如果执行的是JAVA方法,计数器记录正在执行的java字节码地址,如果执行的是native方法,则计数器为空。

虚拟机栈

线程私有的,与线程在同一时间创建。管理JAVA方法执行的内存模型。每个方法执行时都会创建一个桢栈来存储方法的的变量表、操作数栈、动态链接方法、返回值、返回地址等信息。栈的大小决定了方法调用的可达深度(递归多少层次,或嵌套调用多少层其他方法,-Xss参数可以设置虚拟机栈大小)。栈的大小可以是固定的,或者是动态扩展的。如果请求的栈深度大于最大可用深度,则抛出stackOverflowError;如果栈是可动态扩展的,但没有内存空间支持扩展,则抛出OutofMemoryError*(你说看不明白??OK,简单来说就是虚拟机执行Java方法的重要内存模型。内部有一个局部变量表,这个表存储了对象引用,这个局部变量表常被我们成为栈)*。

本地方法区

和虚拟机栈功能相似,但管理的不是JAVA方法,是本地方法,本地方法是用C实现的* 这个对于我们Android程序猿来说基本可以忽略*。

JAVA堆

线程共享的,存放所有对象实例和数组。(比如 AAA a=new AAA(),那么 a这个引用就保存在上面的局部变量表中,而new AAA() 的这个对象就放在堆中

垃圾回收的主要区域。可以分为新生代和老年代(tenured)。新生代用于存放刚创建的对象以及年轻的对象,如果对象一直没有被回收,生存得足够长,老年对象就会被移入老年代。新生代又可进一步细分为eden、survivorSpace0(s0,from space)、survivorSpace1(s1,to space)。刚创建的对象都放入eden,s0和s1都至少经过一次GC并幸存。如果幸存对象经过一定时间仍存在,则进入老年代(tenured)。



方法区

线程共享的,用于存放被虚拟机加载的类的元数据信息:如常量、静态变量、即时编译器编译后的代码。也称为永久代。如果hotspot虚拟机确定一个类的定义信息不会被使用,也会将其回收。回收的基本条件至少有:所有该类的实例被回收,而且装载该类的ClassLoader被回收
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  jvm 内存 java