当前位置: 首页 > news >正文

【JVM】垃圾回收算法

       📝个人主页:五敷有你      

 🔥系列专栏:JVM

⛺️稳中求进,晒太阳

垃圾回收算法

Java是如何实现垃圾回收的呢?简单来说,垃圾回收就做两件事

  1. 找到内存中存活的对象
  2. 释放不在存活对象的内存,使得程序能再次利用这部分空间

垃圾回收算法的历史和分类

1960年发布了第一个GC算法:标记-清除算法。

1963年发布了复制算法

本质上后续所有垃圾回收算法都是在两种算法的基础之上优化而来

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

垃圾回收算法的评价标准

1.吞吐量:

CPU用于执行用户代码的时间与CPU总执行时间的比值,吞吐量数值越高,垃圾回收的效率就越高

即:吞吐量=执行用户代码时间/(执行用户代码时间+GC时间)

2.最大暂停时间

最大暂停时间指的是所有在垃圾回收过程中的STW时间最大值。最大暂停时间越短,用户使用系统时受到的影响越短

3.堆使用的效率

不同的垃圾回收算法,对堆的使用方式是不同的。

上面的三种标准不可兼得。

一般来说,堆内存越大,最大暂停时间就越长,想要减少最大暂停时间,就会降低吞吐量。

标记清除算法

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

1.标记阶段,将所有存活的对象进行标记,Java中使用可达性分析法,从GC ROOT开始通过引用链遍历出所有存活对象。

2.清除阶段,从内存中删除没有被标记也就是非存活的对象,

优点:实现简单,只需要在第一阶段给每个对象维护标志位。第二阶段删除对象即可。

缺点:

1.碎片化问题。

由于内存是连续的,所以在对象被删除后,内存中会出现很多细小的可用内存单元。如果我们需要一个比较大的空间,很可能这些内存单元的大小过小而无法分配。

2.分配速度慢,由于内存碎片的存在,需要维护一个空闲链表,极有可能发生每次需要遍历到链表最后才能获取合适的内存空间

复制算法

复制算法的核心思想是:

  1. 准备两块空间From空间和To空间,每次对象分配阶段,只能使用其中一块空间(From 空间)
  2. 在垃圾回收GC阶段,将From中存活对象赋值到To空间。
  3. 将两块空间的From和To名字互换。

优点:

吞吐量高:复制算法只需要遍历一次存活对象复制到To空间即可,比标记-整理算法少了一次遍历的过程,因而性能较好,但是不如标记-清除算法因为标记清除算法不需要进行对象的移动。

不发生碎片化:复制算法在复制之后就会将对象按顺序放入To空间,所以对象以外 的区域都是可用空间,不存在碎片化内存空间

缺点:

内存使用效率低:每次只能让一半的内存空间来为创建对象使用

标记整理算法

标记整理算法也叫标记压缩算法,是对标记整理算法中容易产生内存碎片问题的一种解决方案。

核心思想分为两个阶段

标记阶段:将所有存活的对象进行标记,Java中使用可达性分析算法,从GC Root开始通过引用链遍历出所有存活对象。

整理阶段:将存活对象移动到堆的一端,清理掉存活对象的内存空间。

分代GC算法

现在优秀的垃圾回收算法,会将上述描述的垃圾回收算法组合进行使用,其中应用最广的就是分代垃圾回收

算法(Generational GC)

分代垃圾回收将整个内存区域划分为年轻代和老年代:

分代回收时,创建出来的对象,首先会被放入Eden伊甸园区

随着对象在Eden区越来越多,如果Eden区满,新创建的对象已经无法放入,就会触发年轻代的GC,称为Minbor GC或者 Young GC。

Minor GC会把需要eden中和From需要回收的对象回收,把没有回收的对象放入To区

接下来S0变成to区,s1变成From区,当eden区满时,在往里面放入对象。依然会发生Minor GC

此时会回收eden区和S1(from)中的对象,并把eden和from区中剩余的对象放入S0

注意:每次Minor GC中都会为对象记录他的年龄,初始值为0,每次GC完成加1

如果Minor GC后对象的年龄达到阈值(最大15,默认值和垃圾回收器有关),对象就会被晋升老年代

当老年代中空间不足,无法放入新的对象时,先尝试minor gc如果还是不足,就会触发Full GC对整个堆进行垃圾回收。

Full GC无法回收老年代的对象,那么当对象继续放入老年代,就会抛出Out Of Memory

http://www.lryc.cn/news/304514.html

相关文章:

  • 如何和将原始request的Header中的值传递给openfeign请求的Header? 以及又如何获取openfeign请求中Header中的值
  • Flink 侧输出流(SideOutput)
  • C语言中关于#include的一些小知识
  • DSP芯片 机器码下载方法 【主要 “扯” 用Uniflash下载的方法】
  • 速盾网络:CDN用几天关了可以吗?安全吗?
  • MR混合现实情景实训教学系统在高空作业课堂中的应用
  • Windows系统中定时执行python脚本
  • HashMap 源码学习-jdk1.8
  • WebStorm 2023:让您更接近理想的开发环境 mac/win版
  • java面试题:数字与字母的映射表
  • Jmeter教程-JMeter 环境安装及配置
  • 十大基础排序算法
  • IP协议及相关技术协议
  • 小红书x-s算法及补环境 单旋转验证码
  • 代码检测规范和git提交规范
  • Elasticsearch:什么是搜索引擎?
  • 人工智能几个关键节点:深蓝,AlphaGo,ChatGPT,Sora
  • WordPres Bricks Builder 前台RCE漏洞复现(CVE-2024-25600)
  • 代码随想录算法训练营总结 | 慢慢总结,想起啥就先写上
  • 基于开源模型对文本和音频进行情感分析
  • SQL中为什么不要使用1=1
  • python 几种常见的音频数据读取、保存方式
  • 关于msvcr120.dll丢失怎样修复的详细解决步骤方法分享,msvcr120.dll文件的相关内容
  • 简单几步通过DD工具把云服务器系统Linux改为windows
  • 使用 package.json 配置代理解决 React 项目中的跨域请求问题
  • 生成 Let‘s Encrypt 免费证书
  • int128的实现(基本完成)
  • 【linux】使用 acme.sh 实现了 acme 协议生成免费的SSL 证书
  • MACOS上面C/C++获取网卡索引,索引获取网卡接口名
  • 解决SSH远程登录开饭板出现密码错误问题