【JVM】垃圾回收

news2024/9/28 19:28:19

6、垃圾回收机制

6.1、对象成为垃圾的判断依据

在堆空间和元空间中,GC这条守护线程会对这些空间开展垃圾回收⼯作,那么GC如何判断这些空间的对象是否是垃圾,有两种算法:

  • 引用计数法:

对象被引用,则计数器+1,如果计数器是0,那么对象将被判定为是垃圾,于是被回收。但是这种算法没有办法解决循环依赖的对象。因此JVM目前的主流⼚商Hotspot没有使⽤这种算法。

  • 可达性分析算法:GC Roots根
    • gc roots根节点: 在对象的引用中,会有这么⼏种对象的变量:来⾃于线程栈中的局部变量表中的变量、静态变量、本地⽅法栈中的变量,这些变量都被称为gc roots根节点
    • 判断依据:gc在扫描堆空间中的某个节点时,向上遍历,看看能不能遍历到gc roots根节点,如果不能,那么意味着这个对象是垃圾。

在这里插入图片描述

6.2、对象中的finalize方法

Object类中有⼀个finalize方法,也就是说任何⼀个对象都有finalize方法。这个方法是对象被回收之前的最后⼀根救命稻草

  • GC在垃圾对象回收之前,先标记垃圾对象,被标记的对象的finalize⽅法将被调用
  • 调用finalize方法如果对象被引用,那么第⼆次标记该对象,被标记的对象将移除出即将被回收的集合,继续存活
  • 调⽤finalize方法如果对象没有被引用,那么将会被回收
  • 注意,finalize方法只会被调用⼀次。

重写Objectfinalize()方法即可

@Override
protected void finalize() throws Throwable {
    if(id%10==0){
        list.add(this);
        System.out.println("对象"+id+"没有被回收");
    }else{
        System.out.println("对象"+id+"被回收");
    }
}

6.3、对象的逃逸分析

在jdk1.7之前,对象的创建都是在堆空间中创建,但是会有个问题,方法中的未被外部访问的对象

public void test1() {
	User user = new User();
    user.setId(1);
	user.setName("xiaoming");
}

public User test2() {
    User user = new User();
    user.setId(1);
    user.setName("xiaoming");
    return user;
}

这种类似于test1()中的user的对象没有被外部访问,且在堆空间上频繁创建,当方法结束,需要被gc,浪费了性能。

所以在1.7之后,就会进行⼀次逃逸分析(默认开启),于是这样的对象就直接在栈上创建,随着⽅法的出栈而被销毁,不需要进行gc。

在栈上分配内存的时候:会把聚合量替换成标量,来减少栈空间的开销,也为了防止步栈上没有⾜够连续的空间直接存放对象。

  • 标量:java中的基本数据类型(不可再分)

  • 聚合量:引用数据类型。

7、垃圾回收算法

7.1、标记清除算法、复制算法、标记整理算法

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

7.2、分代回收算法

在这里插入图片描述

  • 堆空间被分成了新生代(1/3)和老年代(2/3),新生代中被分成了eden(8/10)、survivor1(1/10)、survivor2(1/10)

  • 对象的创建在eden,如果放不下则触发minor gc

  • 对象经过⼀次minorgc 后存活的对象会被放⼊到survivor区,并且年龄+1

  • survivor区执行的复制算法,当对象年龄到达15.进⼊到老年代。

  • 如果老年代放满。就会触发Full GC

7.3、对象进入到老年代的条件

  • 大对象直接进⼊到⽼年代:大对象可以通过参数设置大小,多大的对象被认为是大对象。

-XX:PretenureSizeThreshold

  • 当对象的年龄到达15岁时将进⼊到⽼年代,这个年龄可以通过这个参数设置:

-XX:MaxTenuringThreshold

  • 根据对象动态年龄判断,如果s区中的对象总和超过了s区中的50%,那么下⼀次做复制的时候,把年龄大于等于这次最大年龄的对象都⼀次性全部放⼊到老年代。

  • 老年代空间分配担保机制 :在minor gc时,检查老年代剩余可用空间是否大于年轻代里现有的所有对象(包含垃圾)。如果大于等于,则做minor gc。如果小于,看下是否配置了担保参数的配置:-XX: -HandlePromotionFailure ,如果配置了,那么判断老年代剩余的空间是否小于历史每次minor gc 后进入老年代的对象的平均大小。如果是,则直接full gc,减少⼀次minor gc。如果不是,执行minor gc。如果没有担保机制,直接full gc。

在这里插入图片描述

8、垃圾回收器

垃圾回收机制,我们已经知道什么样的对象会成为垃圾。对象回收经历了什么——垃圾回收算法。那么谁来负责回收垃圾呢?

接下来就来讨论垃圾回收器。

8.1、Serial收集器(-XX:+UseSerialGC -XX:+UseSerialOldGC)

单线程执行垃圾收集,收集过程中会有较长的STW(stop the world),在GC时工作线程不能工作。虽然STW较长,但简单、直接。

新生代采用复制算法,老年代采用标记-整理算法。

在这里插入图片描述

8.2、Parallel收集器(-XX:+UseParallelGC -XX:+UseParallelOldGC)

使用多线程进行GC,会充分利用cpu,但是依然会有stw,这是jdk8默认使用的新⽣代和⽼年代的垃圾收集器。充分利用CPU资源,吞吐量高。

新生代采用复制算法,老年代采⽤标记-整理算法。
在这里插入图片描述

8.3、ParNew收集器(-XX:+UseParNewGC)

工作原理和Parallel收集器⼀样,都是使用多线程进行GC,但是区别在于ParNew收集器可以和CMS收集器配合工作。主流的方案:

ParNew收集器负责收集新生代。CMS负责收集老年代。

在这里插入图片描述

8.4、CMS收集器(-XX:+UseConcMarkSweepGC)

目标:尽量减少stw的时间,提升用户的体验。真正做到gc线程和用户线程几乎同时工作。CMS采用标记-清除算法。

  • 初始标记: 暂停所有的其他线程(STW),并记录gc roots直接能引用的对象。

  • 并发标记:从GC Roots的直接关联对象开始遍历整个对象图的过程, 这个过程耗时较长但是不需要STW,可以与垃圾收集线程⼀起并发运行。这个过程中,用户线程和GC线程并发,可能会导致已经标记过的对象状态发生改变。

  • 重新标记:为了修正并发标记期间因为用户程序继续运行而导致标记产生变动的那⼀部分对象的标记记录,这个阶段的停顿时间⼀般会⽐初始标记阶段的时间稍长,远远比并发标记阶段时间短。主要用到三色标记里的算法做重新标记。

  • 并发清理:开启用户线程,同时GC线程开始对未标记的区域做清扫。这个阶段如果有新增对象会被标记为黑色不做任何处理。

  • 并发重置:重置本次GC过程中的标记数据。

在这里插入图片描述

8.5、三色标记算法

在并发标记阶段,对象的状态可能发⽣改变,GC在进行可达性分析算法分析对象时,用三色来标识对象的状态

  • 黑色:这个对象及其所有引用都已被GC Roots遍历,黑色的对象不会被回收

  • 灰色:这个对象被GC Roots遍历过但其部分的引用没有被GC Roots遍历。在重新标记时重新遍历灰⾊对象。

  • 白色:这个对象没有被GC Roots遍历过。在重新标记时该对象如果是⽩⾊的话,那么将会被回收。

8.6、垃圾收集器组合⽅案

不同的垃圾收集器可以组合使用,在使用时选择适合当前业务场景的组合。

年轻代老年代备注
SerialSerial Old简单、直接
SerialCMS
ParNewCMS推荐使用
ParNewSerial Old
ParallelParallel Old吞吐量⾼、jdk8默认使用的组合
ParallelSerial Old

本文章参考B站 千锋教育JVM全套教程(含jvm调优、jvm虚拟机、jvm面试题、jvm源码详解)系统玩转java虚拟机全程干货无废话,仅供个人学习使用,部分内容为本人自己见解,与千锋教育无关。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/375608.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

搜广推 NeuralCF - 改进协同过滤+矩阵分解的思想

😄 NeuralCF:2017新加坡国立大学提出。【后文简称NCF】 😄 PNN:2016年上海交通大学提出。 文章目录 NeuralCF动机原理general NCFNCF终极版(GMF+MLP的结合)缺点优点ReferenceNeuralCF 动机 前面学了MF,可知MF在用户-物品评分矩阵的基础上做矩阵分解(用户矩阵Q和物品…

Codeforces Round #851 (Div. 2)(A~D)

A. One and Two给出一个数组&#xff0c;该数组仅由1和2组成&#xff0c;问是否有最小的k使得k位置的前缀积和后缀积相等。思路&#xff1a;计算2个数的前缀和即可&#xff0c;遍历判断。AC Code&#xff1a;#include <bits/stdc.h>typedef long long ll; const int N 1…

Maxwell系列:Maxwell采集Mysql到Kafka

目录 Apache Hadoop生态-目录汇总-持续更新 1&#xff1a;直接命令行启动(开发环境使用) 1.1&#xff1a;创建topic&#xff08;可忽略&#xff0c;默认会自动创建&#xff09; 1.2&#xff1a;命令行方式启动maxwell采集通道 1.3&#xff1a;测试流程 2&#xff1a;通过配…

taobao.top.once.token.get( 网关一次性token获取 )

&#xffe5;开放平台免费API必须用户授权聚石塔内调用 网关一次性token获取&#xff0c;对接文档: 公共参数 HTTP地址:http://gw.api.taobao.com/router/rest 公共请求参数: 公共响应参数: 请求参数 响应参数 点击获取key和secret 请求示例 TaobaoClient client new Defa…

2023前端一面vue面试题合集

函数式组件优势和原理 函数组件的特点 函数式组件需要在声明组件是指定 functional:true不需要实例化&#xff0c;所以没有this,this通过render函数的第二个参数context来代替没有生命周期钩子函数&#xff0c;不能使用计算属性&#xff0c;watch不能通过$emit 对外暴露事件&…

【Kafka】生产者

客户端开发 生产者就是负责向kafka发送消息的应用程序。一个正常的生产逻辑的步骤为&#xff1a; 配置生产者客户端参数及创建相应的生产者实例。 构建待发送的消息。 发送消息。 关闭生产者实例。 在创建真正的生产者实例前需要配置相应的参数&#xff0c;比如需要连接的…

顺序表的构造及功能

定义顺序表是一种随机存储都结构&#xff0c;其特点是表中的元素的逻辑顺序与物理顺序相同。假设线性表L存储起始位置为L(A)&#xff0c;sizeof(ElemType)是每个数据元素所占的存储空间的大小&#xff0c;则线性表L所对应的顺序存储如下图。顺序表的优缺点优点&#xff1a;随机…

【面试题】 3 个加强理解TypeScript 的面试问题

TypeScript 作为 JavaScript 的超集&#xff0c;让 JavaScript 越来越像一门“正经的”语言。和纯粹的 JavaScript 语言相比&#xff0c;TypeScript 提供静态的类型检查&#xff0c;提供类的语法糖&#xff08;这一点是继承了 ES6 的实现&#xff09;&#xff0c;让程序更加规范…

Python实现贝叶斯优化器(Bayes_opt)优化LightGBM回归模型(LGBMRegressor算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。1.项目背景贝叶斯优化器 (BayesianOptimization) 是一种黑盒子优化器&#xff0c;用来寻找最优参数。贝叶斯优化器是…

测试用例篇

1.测试用例的意义 测试用例&#xff08;Test Case&#xff09;是为了实施测试而向被测试的系统提供的一组集合&#xff0c;这组集合包含&#xff1a;测试环境、操作步骤、测试数据、预期结果等要素。 测试用例的意义是为了帮助测试人员了解测什么&#xff0c;怎么测 eg&#x…

docker的使用方法

docker技术 同一个操作系统内跑多套不同版本依赖的业务 docker可以使同一个物理机中进程空间&#xff0c;网络空间&#xff0c;文件系统空间相互隔绝 虚拟机弊端&#xff1a;每个需要安装操作系统&#xff0c;太重量级&#xff0c;资源需要提前分配好 部署程序 开发环境 win…

WebRTC现状以及多人视频通话分析

1.WebRTC 概述WebRTC&#xff08;网页实时通信技术&#xff09;是一系列为了建立端到端文本或者随机数据的规范&#xff0c;标准&#xff0c;API和概念的统称。这些对等端通常是由两个浏览器组成&#xff0c;但是WebRTC也可以被用于在客户端和服务器之间建立通信连接&#xff0…

【源码解析】重试模板RetryTemplate源码分析

应用场景 日常开发中&#xff0c;经常会遇到这样的场景&#xff1a;执行一次接口调用&#xff0c;如RPC调用&#xff0c;偶现失败&#xff0c;原因可能是dubbo超时、连接数耗尽、http网络抖动等&#xff0c;出现异常时我们并不能立即知道原因并作出反应&#xff0c;可能只是一…

新冠小阳人症状记录

原想挺过春节后再养&#xff0c;发现事与愿违。生理期期间抵抗力下降&#xff0c;所以在生理期第二天就有些症状了。可能是生理期前一天出去采购食物染上&#xff0c;也可能是合租夫妻染上。anyway&#xff0c;记录下自己的症状与相应有效的偏方&#xff1a; 第一天&#xff1a…

git构建工具

目录 一、简介 1、版本控制 2、版本控制工具 二、Git工作机制 1、工作区 2、暂存区 3、远程库 三、常用命令 1、初始化本地仓库 git init 2、查看本地库状态 git status 3、工作区文件加入到暂存区 git add 4、 暂存区文件提交到本地库 git commit -m &quo…

Odoo | Webserivce | 5分钟学会【JSONRPC】接口开发

文章目录Odoo - JsonRPC1. Odoo内方法结构&#xff08;接收端&#xff09;2. POST接口请求结构&#xff08;发送端&#xff09;3. 实例测试Odoo - JsonRPC 1. Odoo内方法结构&#xff08;接收端&#xff09; # -*- coding: utf-8 -*- import odoo import logging import trac…

手把手带你做一套毕业设计-征程开启

本文是《手把手带你做一套毕业设计》专栏的开篇&#xff0c;文本将会包含我们创作这个专栏的初衷&#xff0c;专栏的主体内容&#xff0c;以及我们专栏的后续规划。关于这套毕业设计的作者呢前端部分由狗哥负责&#xff0c;服务端部分则由天哥操刀。我们力求毕业生或者新手通过…

Anaconda3 +pycharm详细安装教程(2023年)

前言最近配置了一台新电脑&#xff0c;准备安装Anaconda&#xff0c;原来是直接安装的python安装包以及pycharm&#xff0c;需要使用什么包就安装什么包&#xff0c;由于网络原因&#xff0c;经常安装失败&#xff0c;所以选择包含众多科学数据包的Anaconda。说到这里&#xff…

@font-face用法超详细讲解

文章目录font-face是什么font-face基本语法urlTTFOTFEOTWOFFSVGformatfont-face用法示例font字体下载ttf-to-eot 字体转换器https://blog.csdn.net/qq_37417446/article/details/106728725 https://developer.mozilla.org/zh-CN/docs/Web/CSS/font-face font-face是什么 font-…

网络性能总不好?专家帮你来“看看”— CANN 6.0 黑科技 | 网络调优专家AOE,性能效率双提升

随着深度学习模型复杂度和数据集规模的增大&#xff0c;计算效率的提升成为不可忽视的问题。然而&#xff0c;算法网络的多样性、输入数据的不确定性以及硬件之间的差异性&#xff0c;使得网络调优耗费巨大成本&#xff0c;即使是经验丰富的专家&#xff0c;也需要耗费数天的时…