JVM规范&JVM优化

JVM内存结构 & Java垃圾回收机制(GC)

2019-01-24  本文已影响0人  左大人

序言

相信各位Android和Java开发的同学都知道,Java与C++不同,JVM会自动管理内存,即自动帮我们分配内存和回收内存,而垃圾回收机制(GC)就是用来帮助我们回收内存的。
那么,问题就来了,垃圾回收机制会涉及到哪些内存区域,以及回收的过程如何?这就需要我们对JVM内存结构有一个清晰的认识。
下面,我们先介绍JVM内存结构,再介绍垃圾回收机制。

JVM内存结构

请看下图:


image.png

图中清晰的划分了JVM内存结构,针对上图,做一些简要的说明:

下面,将对每个区域进行详细的介绍。

1. 方法区

别名:持久代
作用:主要用来存储虚拟机加载的类结构信息(版本、字段、方法、接口)、常量、静态变量、还有及时编译后的代码
特点

2. 运行时常量池

作用:主要用于存储Java类文件常量池中的符号信息
特点

3. 堆

别名:Java堆、GC堆
作用:用来存储对象实例和数组(简单来说所有new出来的对象都存在堆内存中)
特点

Generation
通过下图简单认识新生代、老年代和持久代。

image.png

4. 程序计数器

作用:保证在多线程环境中程序可以连续执行。存放当前线程执行字节码的行号,字节码解释器工作时,是通过改变计数器的值来选取下一条字节码指令。
特点

5. JVM栈

作用:用来描述方法执行的内存区域。
特点

栈的组成
栈是用来存储栈帧的,每当线程调用一个方法,就会产生一个栈帧,压入栈内。
而栈帧是由局部变量表、操作数栈和帧数据区组成:

6. 本地方法栈

跟JVM栈类似,只不过JVM栈用来执行Java方法,而本地方法栈用来执行Native方法。

到这里,JVM内存结构介绍完毕,有了这个基础,下面我们一起学习Java垃圾回收机制。

Java垃圾回收机制(GC)

首先,我们需要有个概念,垃圾回收机制主要是回收堆内存区域,偶尔也会回收方法区。
关于垃圾回收,我们需要弄清楚下面三个问题:

带着这些问题,我们来展开学习。

关于垃圾回收机制,必须记住一规则:Stop-the-World(JVM执行GC时线程都会处于等待状态,任何GC算法都是如此)

1. 确定对象是否可以回收

这里有两种经典算法确定对象是否可以被回收:引用计数算法可达性分析算法

引用计数算法
简介:根据对象被引用的数量来判断对象是否可以被回收。
详细描述:这是垃圾收集器早期的一种策略,堆中每个对象实例都有一个引用计数器。对象A,当它的引用赋值给一个变量(如a = A,b = a),则引用计数器+1;当A的引用变量生命周期结束或者设置一个新值(如a = B),那么引用计数器-1。特殊情况:当一个对象实例被垃圾收集器回收,该对象引用的任何实例的引用计数器都-1
优点:引用计数收集器执行速度很快,不会长时间打断程序的执行。
缺点:很难解决对象之间相互循环引用问题。

image.png

这个时候,对象A和对象B都无法被回收。

可达性分析算法
简介:根据对象引用链是否可达来判断对象是否可以被回收。
详细描述:程序把所有的引用关系看做一张拓扑图,通过一系列的"GC Roots"作为起点,这些节点向下索引,搜索所走过的路径称为引用链,当一个对象没有任何引用链到达"GC Roots",那么这个对象不可达,可以被回收。可以作为"GC Roots"的对象包括:

2. 垃圾回收时机

垃圾回收有两种类型:MinorGC和FullGC
MinorGC
对新生代进行回收,不影响老年代,因为新生代对象大多死亡频繁,所以MinorGC也会频发触发。

FullGC
也叫MajorGC,对整个堆进行回收,包括新生代和老年代,由于回收范围大,所以速度慢,因此要尽量少触发FullGC。

对于不同的垃圾收集器,MinorGC和FullGC的触发时机也不一样,我们就以HotSpot的serial GC实现来看:

3. 垃圾收集算法

经典的垃圾回收算法有**标记清除算法、复制算法、标记整理算法、分代收集算法。

标记清除算法
简介:从根集合进开始扫描,标记存活的对象。再扫描整个空间中没有被标记的对象,进行回收。
详细描述:标记和清除两个过程的效率都不高,该算法不需要对对象进行移动,仅清除未标记的对象,清除之后,容易产生大量不连续的内存碎片。程序运行过程中,需要分配较大的对象时,无法找到足够的连续的内存,可能不得不触发另一次垃圾回收操作。

image.png

复制算法
简介:把可用的内存空间划分为大小相同的两块,每次只使用其中一块,当这一块用完之后,就把存活的对象复制到另外一块空间,把当前内存空间一次性清理掉。
详细描述:适用于对象存活率低的场景,如新生代。由于每次都对整个半区进行回收,不用考虑内存碎片,只要移动堆顶指针,按顺序分配内存即可,实现简单高效。事实上,商用虚拟机都采用这种方式回收新生代,据统计,新生代每次回收大概只有10%的存活对象。

image.png

标记整理算法
简介:这是标记清除算法的改进版本,标记过程相同,不过后续让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存。
详细介绍:类似于磁盘整理过程,适用于存活率高的老年代。对于老年代,复制算法效率会变低,因为对象存活率高,每次都要对很多对象进行复制操作。

image.png

分代收集算法
简介:不同生命周期的对象位于堆中不同区域,生命周期短位于新生代,生命周期长位于老年代,不同区域采取不同的回收策略,提高JVM执行效率。
详细描述:当代商用虚拟机都采用了分代收集算法,新生代采用复制算法,老年代采用标记整理算法(或标记清除算法)

四种引用对GC的影响

我们知道,Java中有四种引用方式:强引用、软引用、弱引用、虚引用

总结

到这列,本文内容介绍完毕,我们来总结一下:
本文先介绍了JVM内存结构,主要分为方法区、堆、栈和程序计数器。
了解JVM内存结构之后,我们继续介绍垃圾回收机制,主要从怎样确定对象是否能够被回收、回收时机、回收算法这三个方面进行介绍。
最后,讲解了Java四种引用方式对GC的影响。
看到这里的小伙伴,希望你有所收获!!

上一篇下一篇

猜你喜欢

热点阅读