| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> Java知识库 -> JVM 垃圾回收 -> 正文阅读 |
|
[Java知识库]JVM 垃圾回收 |
JVM 垃圾回收1. 如何判断对象可以回收1.1 引用计数法定义:引用计数法(Reference Counting)给对象中添加一个引用计数器,每当有一个地方引用它时,计数器值就加1;当引用失效时,计数器值就减1;任何时刻计数器为 0 的对象就是不可能再被使用的。 优点:实现简单,判定效率高。 1.2 可达性分析算法定义:可达性分析法(Reachability Analysis),通过一系列的称为 “GC Roots” 的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链(Reference Chain),当一个对象到 GC Roots 没有任何引用链相连接(用图论的话来说,就是从 GC Roots 到这个对象不可达)时,则证明此对象是不可用的。 JVM 虚拟机中的垃圾回收器采用可达性分析来探索所有存活的对象。扫描堆中的对象,看能否沿着 GC Roots 对象为起点的引用链找到该对象,如果找不到,则表示可以回收。 哪些对象可以作为 GC Roots?
1.3 五种引用1.3.1 强引用定义:指在程序代码中普遍存在的,类似 “Object obj = new Object()“ 这类的引用,只要强引用还存在,垃圾收集器永远不会回收掉被引用的对象。
1.3.2 软引用定义:用来描述一些还有用但并非必需的对象。对于软引用关联着的对象,在系统将要发生内存溢出异常之前,将会把这些对象列进回收范围之中进行第二次回收。如果这次回收还没有足够的内存,才会抛出内存溢出异常。
软引用的使用
如果在垃圾回收时发现内存不足,在回收软引用所指向的对象时,软引用本身不会被清理。 如果想要清理软引用,需要使用引用队列:
思路:查看引用队列中有无软引用,如果有,则将该软引用从存放它的集合中移除(这里为一个 list 集合)
1.3.3 弱引用定义:弱引用用来描述非必需对象,但是它的强度比软引用更弱一些,被弱引用关联的对象只能生存到下一次垃圾收集之前。
弱引用的使用和软引用类似,只是将 SoftReference 换为了 WeakReference 1.3.4 虚引用定义:虚引用也称为幽灵引用或者幻影引用,它是最弱的一种引用关系。虚引用不会对对象的生存时间构成任何影响,也无法通过虚引用来取得一个对象实例。设置虚引用的目的是为了能够在这个对象被收集器回收时收到一个系统通知。
1.3.5 终结器引用所有的类都继承自 Object 类,Object 类有一个 finalize 方法。当某个对象不再被其他的对象所引用时,会先将终结器引用对象放入引用队列中,然后根据终结器引用对象找到它所引用的对象,然后调用该对象的 finalize 方法,第二次 GC 时才能回收被引用对象。 1.4 引用队列软引用和弱引用可以配合引用队列:
虚引用和终结器引用必须配合引用队列:
1.5 生存还是死亡
2. 垃圾回收算法2.1 标记-清除算法定义:标记-清除(Mark-Sweep)算法,顾名思义分为 ”标记“ 和 ”清除“ 两个阶段,是指在虚拟机执行垃圾回收的过程中,首先采用标记算法确定可回收对象,在标记完成后统一回收所有被标记的对象,为堆内存腾出相应的空间。
优点:思路简单,容易实现。 缺点:
2.2 复制算法定义:复制(Copying)算法,将内存分为等大小的两个区域,FROM 和 TO(TO 中为空)。先将被 GC Roots 引用的对象从FROM 放入 TO 中,再回收不被 GC Roots 引用的对象,最后交换 FROM 和 TO. 这样也可以避免内存碎片的问题,但是会占用双倍的内存空间。
缺点:内存缩减为原来的一半,内存利用率仅为 50% 2.3 标记-整理算法定义:标记-整理(Mark-Compact)算法和标记-清除算法非常的类似,主要被应用于老年代中,可分为以下两步:
但是同样,标记-整理算法也有它的缺点,一方面它要标记所有存活对象,另一方面还添加了对象的移动操作以及更新引用地址的操作,因此标记-整理算法具有更高的使用成本。 3. 分代垃圾回收定义: 根据对象存活周期的不同将内存划分为几块,一般是把 Java 堆分为新生代和老年代,这样就可以根据各个年代的特点采用最适当的收集算法。在新生代中,每次垃圾收集时都有大批对象死去,只有少量存活,则选用复制算法,只需要付出少量存活对象的复制成本就可以完成收集。而老年代中因为对象存活率高、没有额外空间对它进行分配担保,则必须使用”标记-清理“或者”标记-整理“算法来进行回收。 新创建的对象都被放在了新生代的伊甸园中。 Minor GC 会将伊甸园和幸存区 FROM 存活的对象先复制到幸存区 TO 中, 并让其寿命加 1,再交换两个幸存区。再次创建对象,若新生代的伊甸园又满了,则会再次触发 Minor GC(会触发 stop the world, 暂停其他用户线程,只让垃圾回收线程工作),这时不仅会回收伊甸园中的垃圾,还会回收幸存区中的垃圾,再将活跃对象复制到幸存区 TO 中。回收以后会交换两个幸存区,并让幸存区中的对象寿命加 1. 归纳:
GC 分析:
相关 VM 参数:
4. 垃圾回收器
并行(Parallel)收集:指多条垃圾收集线程并行工作,但此时用户线程仍处于等待状态。 并发(Concurrent)收集:指用户线程与垃圾收集线程同时工作(不一定是并行的可能会交替执行)。用户程序在继续运行,而垃圾收集程序运行在另一个 CPU 上。 吞吐量:即 CPU 用于运行用户代码的时间与 CPU 总消耗时间的比值(吞吐量 = 运行用户代码时间 / ( 运行用户代码时间 + 垃圾收集时间 ))。例如:虚拟机共运行 100 分钟,垃圾收集器花掉1分钟,那么吞吐量就是 99%. 1、串行
2、吞吐量优先
3、响应时间优先
4.1 串行开启串行垃圾回收 JVM 命令参数: 因为是串行的,所以只有一个垃圾回收线程。且在该线程执行回收工作时,其他线程进入阻塞状态。 Serial 收集器 Serial 收集器是最基本的、发展历史最悠久的收集器。 特点:单线程、简单高效(与其他收集器的单线程相比),新生代采用复制算法暂停所有用户线程,老年代采取标记-整理算法暂停所有用户线程。对于限定单个 CPU 的环境来说,Serial收集器由于没有线程交互的开销,专心做垃圾收集自然可以获得最高的单线程收集效率。收集器进行垃圾回收时,必须暂停其他所有的工作线程,直到它结束(Stop The World)。 ParNew 收集器 ParNew 收集器其实就是 Serial 收集器的多线程版本。这款收集器是 HotSpot 虚拟机中第一款真正意义上的并发(Concurrent)收集器,它第一次实现了让垃圾收集线程与用户线程(基本上)同时工作。 特点:多线程、ParNew 收集器默认开启的收集线程数与 CPU 的数量相同,在 CPU 非常多的环境中,可以使用 Serial Old 收集器 Serial Old 是 Serial 收集器的老年代版本。 特点:同样是单线程收集器,采用标记-整理算法。 4.2 吞吐量优先吞吐量就是 CPU 用于运行用户代码的时间与 CPU 总消耗时间的比值,即吞吐量=运行用户代码时间 / (运行用户代码时间+垃圾收集时间)。
Parallel Scavenge 收集器 与吞吐量关系密切,故也称为吞吐量优先收集器。 特点:属于新生代收集器也是采用复制算法的收集器(用到了新生代的幸存区),又是并行的多线程收集器(与 ParNew 收集器类似)。 该收集器的目标是达到一个可控制的吞吐量。还有一个值得关注的点是:GC 自适应调节策略(与 ParNew 收集器最重要的一个区别)。 GC自适应调节策略:Parallel Scavenge 收集器可设置 -XX:+UseAdptiveSizePolicy 参数。当开关打开时不需要手动指定新生代的大小(-Xmn)、Eden 与 Survivor 区的比例(-XX:SurvivorRation)、晋升老年代的对象年龄(-XX:PretenureSizeThreshold)等,虚拟机会根据系统的运行状况收集性能监控信息,动态设置这些参数以提供最合适的停顿时间和最大的吞吐量,这种调节方式称为GC的自适应调节策略。 Parallel Scavenge 收集器使用两个参数控制吞吐量:
Parallel Old 收集器 特点:多线程,采用标记-整理算法(老年代没有幸存区)。 4.3 响应时间优先CMS 收集器 CMS(Concurrent Mark Sweep) 收集器,是一种以获取最短回收停顿时间为目标的老年代收集器。
优点:并发收集、低停顿,响应速度快。 缺点:
应用场景:适用于注重服务的响应速度,希望系统停顿时间最短,给用户带来更好的体验等场景下。如 Web 程序、B/S 服务。 CMS 收集器的运行过程分为下列 4 步:
并发标记和并发清除过程垃圾收集器线程都可以与用户线程工作,总体上说CMS 收集器的内存回收过程是与用户线程一起并发执行的。 4.4 G1定义:G1(Garbage First) 垃圾收集器,
相关 JVM 参数:JDK 8 并不是默认开启的,需要参数开启:
G1 垃圾回收阶段: 4.4.1 Young Collection分区算法 Region 分代是按对象的生命周期划分,分区则是将堆空间划分连续几个不同小区间,每一个小区间独立回收,可以控制一次回收多少个小区间,方便控制 GC 产生的停顿时间。 E:伊甸园 S:幸存区 O:老年代
4.4.2 Young Collection + CMCM:Courrent Mark,并发标记
4.4.3 Mixed Collection会对 E S O 进行全面的回收:
JVM 参数 问:为什么有的老年代被拷贝了,有的没拷贝? 因为指定了最大停顿时间,如果对所有老年代都进行回收,耗时可能过高。为了保证时间不超过设定的停顿时间,会回收最有价值的老年代(回收后,能够得到更多内存)。 4.4.4 Full GCG1 在老年代内存不足时(老年代所占内存超过阈值)
4.4.5 Young Collection 跨代引用新生代回收的跨代引用(老年代引用新生代)问题 卡表与 Remembered Set
在引用变更时通过 post-write barried + dirty card queue concurrent refinement threads 更新 Remembered Set 4.4.6 Remark重新标记阶段,发生在垃圾回收时,收集器处理对象的过程中。 黑色:已被处理,需要保留的,也就是说不会被回收;灰色:正在处理中的;白色:还未处理的,有可能以后被回收
4.4.7 JDK 8u20 字符串去重优点:节省大量内存
4.4.8 JDK 8u40 并发标记类卸载所有对象都经过并发标记后,就能知道哪些类不再被使用,当一个类加载器的所有类都不再使用,则卸载它所加载的所有类。 4.4.9 JDK 8u60 回收巨型对象
G1 会跟踪老年代所有 incoming 引用,如果老年代 incoming 引用为 0 的巨型对象就可以在新生代垃圾回收时处理掉。 4.4.10 JDK 9 并发标记起始时间的调整
|
|
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 | -2024/11/24 4:46:12- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |