JVM内存模型&回收算法

内存结构

JVM分区的意义在于,JVM定义了不同运行阶段的数据区。
某些区域随着JVM启动及销毁,另外一些区域的数据随着线程创建和销毁。
如下图:
这里写图片描述

程序计数器(The Program Counter Register)是线程私有的内存,每条线程都有一个独立的程序计数器,用于CPU切换线程时记录当前线程将要执行的下一条字节码行号。

虚拟机栈 (Java Virtual Machine Stacks) 就是所说的“栈内存”
● 是线程私有的内存,生命周期与线程相同,用于管理JAVA方法执行的内存模型。
● 每个方法执行时都会创建一个桢栈来存储方法的私有变量、操作数栈、动态链接方法、返回值、返回地址等信息。
● 栈的大小决定了方法调用的可达深度(递归多少层次,或嵌套调用多少层其他方法,-Xss参数可以设置虚拟机栈大小)。
● 栈的大小可以是固定的,或者是动态扩展的。如果栈的深度是固定的,请求的栈深度大于最大可用深度,则抛出stackOverflowError;如果栈是可动态扩展的,但没有内存空间支持扩展,则抛出OutofMemoryError。
这里写图片描述

本地方法区(Native Method Stacks)
与虚拟机栈类似,但是管理的是本地方法,用C实现的。

以上三个区域都是线程私有的,而下面两个区域都是线程共享的。
这里写图片描述

JAVA堆(JVM heap)就是所说的“堆内存”,体积最大
● 用于存放对象实例和数组,是GC的主要关注区域。
● 可以分为新生代和老年代。新生代又可进一步细分为eden、survivorSpace0(s0,from space)、survivorSpace1(s1,to space)。

这里写图片描述

方法区( Method Area) 是线程共享的
● 用于存放被虚拟机加载的类信息:如常量、静态变量、即时编译器编译后的代码。
● 也称为“永久代”,但是也会被回收。
● 回收的基本条件至少有:所有该类的实例被回收,而且装载该类的ClassLoader被回收

垃圾回收算法
● 标记-清除算法(Mark-Sweep)
从根节点开始标记所有可达对象,其余没标记的即为垃圾对象,执行清除。但回收后的空间是不连续的。

● 复制算法(copying)
将内存分成两块,每次只使用其中一块,垃圾回收时,将标记的对象拷贝到另外一块中,然后完全清除原来使用的那块内存。
复制后的空间是连续的。复制算法适用于新生代,因为垃圾对象多于存活对象
在新生代串行垃圾回收算法中,将eden中标记存活的对象拷贝未使用的s1中,s0中的年轻对象也进入s1,如果s1空间已满,则进入老年代;这样交替使用s0和s1。这种改进的复制算法,既保证了空间的连续性,有避免了大量的内存空间浪费。
这里写图片描述
● 标记-压缩算法(Mark-compact)
适合用于老年代的算法(存活对象多于垃圾对象)。
标记后不复制,而是将存活对象压缩到内存的一端,然后清理边界外的所有对象。
这里写图片描述
参考:https://blog.csdn.net/kingofworld/article/details/17718587

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值