JVM深入原理(八)(二):垃圾回收

 JVM系列文章

  1. 深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器_eden used total max-优快云博客
  2. JVM深入原理(一+二):JVM概述和JVM功能-优快云博客
  3. JVM深入原理(三+四):JVM组成和JVM字节码文件-优快云博客
  4. JVM深入原理(五):JVM组成和JVM字节码文件-优快云博客
  5. JVM深入原理(六)(一):JVM类加载器-优快云博客
  6. JVM深入原理(六)(二):双亲委派机制-优快云博客
  7. JVM深入原理(七)(一):运行时数据区-优快云博客
  8. JVM深入原理(七)(二):运行时数据区-优快云博客
  9. JVM深入原理(八)(一):垃圾回收-优快云博客
  10. JVM深入原理(八)(二):垃圾回收-优快云博客

目录

8.6. 垃圾回收-算法

8.6.1. 算法评判标准

8.6.2. 标记-清除算法

8.6.3. 标记-整理算法

8.6.4. 复制算法

8.6.5. 分代垃圾回收算法

8.6.5.1. 分代垃圾回收算法-结构

8.6.5.2. 对象存储和垃圾回收过程

8.6.5.3. 相关JVM参数

8.7. 垃圾回收器

8.7.1. 串行垃圾回收器

8.7.2. 吞吐量优先垃圾回收器

8.7.3. 响应时间优先垃圾回收器

8.7.4. G1

8.7.4.1. G1垃圾回收阶段

8.7.4.2. Young Collection

8.7.4.3. Young Collection + Concurrent Mark

8.7.4.4. Mixed Collection


8.6. 垃圾回收-算法

8.6.1. 算法评判标准

Java垃圾回收过程会通过单独的GC线程来完成,但是不管使用哪一种GC算法,都会有部分阶段需要停止所有的用户线程。这个过程被称之为StopTheWorld简称STW,如果STW时间过长则会影响用户的使用。

评判GC算法从三个维度:

  1. 吞吐量:吞吐量越大说明GC算法效率越高,吞吐量=(代码执行时间)/(代码执行时间+GC时间)
  2. 最大暂停时间:最大STW时间越小效率越高,最大暂停时间指的是所有垃圾回收过程STW的最大值
  3. 堆的使用效率:不同的垃圾回收算法堆堆的使用效率是不同的,标记-整理算法可以使用完整的堆,而复制算法将堆一分为二,只能使用一半的堆

一般来说,堆内存越大,最大STW就越长,想减少最大STW,就会减少吞吐量,不同的GC算法适用于不同的场景

8.6.2. 标记-清除算法

标记清除算法分为两个阶段:

  1. 标记:先沿着GC Roots对象的引用链寻找未被引用的对象,标记起来
  2. 清除:将未被引用的对象垃圾回收掉

标记清除算法的优缺点:

  • 优点:效率高速度快
  • 缺点:容易产生内存碎片,分配新的对象时需要遍历整个空间才能得出是否有空间分配

8.6.3. 标记-整理算法

标记整理算法

  • 介绍:在标记清除算法的基础上多一步整理内存空间过程
  • 优点:不会产生内存碎片
  • 缺点:移动对象会改变所有引用了该地址的对象,速度慢

8.6.4. 复制算法

复制算法

  • 介绍:开辟两块一模一样的内存空间,先标记未应用的对象后,将已引用的对象复制到未使用的内存空间,清空原来空间,再交换两者的位置
  • 优点:不会产生内存碎片
  • 缺点:占用双倍内存空间

8.6.5. 分代垃圾回收算法

8.6.5.1. 分代垃圾回收算法-结构

分代回收算法将整个堆中的区域划分为新生代和老年代

  • 新生代:存放存活时间短的对象
    • eden区:
    • survivor s0
    • survivor s1
  • 老年代:存放存活时间长的对象

8.6.5.2. 对象存储和垃圾回收过程
  1. 对象首先分配在eden区中
  2. eden区内存不足时会触发minor gc,eden区和survivor From区存活的对象使用复制算法复制到survivor to区中,存活对象寿命+1,并交换From和To区,以此往复
  3. minor gc会触发一次stop the world,暂停其他用户线程,等待垃圾回收完毕,用户线程才恢复运行
  4. 当对象寿命达到阈值时会转移到老年代中,最大寿命是15
  5. 老年代内存不足时,会先尝试触发minor gc,如果内存仍不足,会触发full gc,stw时间更长,采用的是标记整理算法

--超过新生代大小的大对象会直接晋升到老年代

8.6.5.3. 相关JVM参数

8.7. 垃圾回收器

8.7.1. 串行垃圾回收器

  • 特点:单线程,垃圾回收时其他线程都暂停
  • 使用场景:堆内存较小,单核CPU
  • 流程图:serial新生代(复制)+serialOld(标记整理)

8.7.2. 吞吐量优先垃圾回收器

  • 特点:多线程,尽可能让每次stop the world回收的垃圾数量多,类似多食少餐,并行,多个CPU运行单个垃圾回收线程
  • 使用场景:堆内存较大,多核CPU
  • 流程图:

    • -XX:+UseParallelGC~-XX:+UseParallel0ldGC:开启并行垃圾回收器,这个参数同时影响新生代和老年代,JDK8下默认开启
    • -XX:+UseAdaptiveSizePolicy:采用动态自适应新生代大小空间分配
    • -XX:GCTimeRatio=ratio:调整吞吐量大小,垃圾回收时间=1/(1+radio)
    • -XX:MaxGCPauseMillis=ms:调整暂停时间
    • -XX:ParallelGCThreads=n:控制parallelgc工作时的线程数

8.7.3. 响应时间优先垃圾回收器

  • 特点:多线程,尽可能让每次stop the world的时间尽可能短,类似少食多餐,并发,单个CPU运行垃圾回收线程和用户线程,只有在并发gc清理垃圾速度跟不上垃圾产生速度的时候才会产生full gc
  • 使用场景:堆内存较大,多核CPU
  • 流程图:

    • -XX:+UseConcMarkSweepGC~-XX:+UseParNewGC~SerialOld
    • XX:ParallelGCThreads=n~-XX:ConcGCThreads=threads-
    • XX:CMSInitiatingOccupancyFraction=percent-
    • XX:+CMSScavengeBeforeRemark

8.7.4. G1

  • 定义:garbage first,只有在并发gc清理垃圾速度跟不上垃圾产生速度的时候才会产生full gc
  • 适用场景:
    • 同时注重吞吐量和低响应时间,默认STW时间是200ms
    • 超大堆内存,会将堆划分为多个大小相等的区域
    • 整体上是标记整理算法,两个区域之间是复制算法
  • 相关JVM参数
    • -XX:+UseG1GC:JDK8以前开启G1垃圾回收器
    • -XX:G1HeapRegionSize=size
    • -XX:MaxGCPauseMillis=time
8.7.4.1. G1垃圾回收阶段

8.7.4.2. Young Collection

  • 初始创建的对象存放在eden区
  • 经历一次垃圾回收后对象存放在survivor区
  • 对象寿命达到阈值后经过复制算法存放在老年代
8.7.4.3. Young Collection + Concurrent Mark

  • Young GC时会进行GC Root的初始标记
  • 老年代占用堆空间的比例达到阈值时,会触发并发标记
  • -XX:InitiatingHeapOccupancyPercent=percent(默认45%):调整阈值
8.7.4.4. Mixed Collection

  • 会对eden区,survivor区,老年代的对象进行full gc
  • 优先回收垃圾最多的区域,保证一次回收的垃圾释放最多的空间
  • 最终标记会stw
  • 拷贝存货会stw
  • -XX:MaxGCPauseMillis=ms
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值