[关闭]
@lzb1096101803 2016-03-12T12:19:39.000000Z 字数 1067 阅读 364

JVM内存区域和堆分区

电话面试


内存区域

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

Java虚拟机栈:线程私有,局部变量表(基本数据类型,对象引用),操作数栈,动态链接,方法出口
Java堆:分配对象,线程共享
方法区:存储被加载的类信息,常量,静态变量,编译后的代码
运行时常量池:存放变异生成的字面量和符号引用
本地方法栈
直接内存

堆分区

Eden,survival from to,老年代,Perm/Metaspace
8:1:1

某一时刻只有其中一个是被使用的,另外一个留做垃圾收集时复制对象用,在Young区间变满的时候,minor GC就会将存活的对象移到空闲的Survivor区间中,根据JVM的策略,在经过几次垃圾收集后,任然存活于Survivor的对象将被移动到Tenured区间。

Tenured区主要保存生命周期长的对象,一般是一些老的对象,当一些对象在Young复制转移一定的次数以后,对象就会被转移到Tenured区,一般如果系统中用了application级别的缓存,缓存中的对象往往会被转移到这一区间。

Perm代主要保存class,method,filed对象,这部门的空间一般不会溢出,除非一次性加载了很多的类,不过在涉及到热部署的应用服务器的时候,有时候会遇到java.lang.OutOfMemoryError : PermGen space 的错误,造成这个错误的很大原因就有可能是每次都重新部署,但是重新部署后,类的class没有被卸载掉,这样就造成了大量的class对象保存在了perm中,这种情况下,一般重新启动应用服务器可以解决问题。

纳入字符串常量池的字符串被存储在永久代中,永久代清除废弃常量和无用的类(3个条件,类实例回收,ClassLoader回收,对应的Class对象没被引用),而且只是可以。

但是这并不意味着类的元数据信息也消失了。这些数据被移到了一个与堆不相连的本地内存区域,这个区域就是我们要提到的元空间。,因为对永久代进行调优是很困难的。永久代中的元数据可能会随着每一次Full GC发生而进行移动。并且为永久代设置空间大小也是很难确定的,因为这其中有很多影响因素,比如类的总数,常量池的大小和方法数量等。永久代的垃圾回收和老年代的垃圾回收是绑定的,一旦其中一个区域被占满,这两个区都要进行垃圾回收。

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注