02JVM_垃圾回收GC

二、垃圾回收GC

里面存放着java的所有对象实例,当对象为“死去”,也就是不再使用的对象,就会进行垃圾回收GC

1.如何判断对象可以回收

1.1引用计数器

介绍

在对象中添加一个引用计数器,当一个对象被其他变量引用时这个对象的引用计数器加1。当某个变量不再引用这个对象时引用计数器减1。当这个引用计数变为0时,这个对象就会被垃圾回收。

优点

判定效率很高

缺点

不完全准确,当两个对象相互引用就判断失效了。当A引用B对象,B引用A对象,然后A和B不再被其他变量引用,垃圾不会回收,出现了垃圾回收失效。

1.2可达性分析算法(Java)

介绍

①要确定一个根对象GC Roots(肯定不会被垃圾回收的对象)作为起始节点,当垃圾回收前会对堆对象进行扫描,判断这些对象是否被根对象引用,如果没有被引用那么这个对象就可以垃圾回收。

②java虚拟机是通过可达性分析判断存活对象

哪些对象可以作为GC Root?

①虚拟机栈中引用的对象

各个线程调用的方法(参数,局部变量)

②本地方法栈中native方法引用的对象

③方法区中类静态属性引用的对象

Java类的引用类型静态变量

④方法区中常量引用的对象

字符串常量池StringTable里的引用

⑤所有被同步锁synchronized持有的对象

⑥java虚拟机内部的引用,核心类

1.3四种引用

1.强引用-不回收

介绍

程序代码中普遍存在的引用赋值“Object obj = new Object()”这种引用关系。

特点

不回收,只要沿着GC Root引用链能够找到这个对象,这个对象就不会被垃圾回收。

2.软引用-内存不足回收

介绍

软引用(间接引用)是一些还有用但非必需的对象,当被软引用关联的对象,系统发生内存溢出钱,会把这些对象列进回收范围进行二次回收。如果这次回收没有足够的内存,会抛出内存溢出异常

特点

当垃圾回收后,此时内存仍不够,软引用关联的对象会进行垃圾回收(内存不够才回收)

3.弱引用-发现就回收

介绍

弱引用是一些还有用但非必需的对象,生存到下一次垃圾回收为止。在系统进行垃圾回收时,发现弱引用,不管系统堆空间是否充足,都会回收软引用关联的对象。

特点

当发生垃圾回收时,弱引用关联的对象都会被回收。(内存不管够不够都回收)

4.虚引用-对象回收跟踪

介绍

当虚引用对象创建时会关联一个引用队列。虚引用在创建时必须提供引用队列作为参数。当垃圾回收准备回收一个对象时,如果发现他是虚引用,在垃圾回收后把这个虚引用加入引用队列,通知应用程序对象的回收情况。

5.终结器引用-引用队列配合

介绍

用于实现对象的finalize()方法。当终结器对象创建时会关联一个引用队列。当准备回收一个对象时,发现是终结器引用,会把终结器引用对象放入引用队列(此时对象没回收)。由Finalizer线程调用该对象的finalize()方法,第二次垃圾回收就会回收此对象。

2.垃圾回收算法

2.1标记清除

什么时候垃圾回收

扫描整个堆对象的过程中,如果发现对象被GC Root引用了,需要保留。如果这个对象没有被GC Root引用,就要进行垃圾回收。

标记清除算法的两个阶段

①把没有引用的对象标记为垃圾。

②把垃圾对象占用的空间释放,清除

③释放不是空间清0,而是把起始地址记录下来,然后将来把创建的新对象存储这个位置

优点

垃圾回收速度快

缺点

产生内存碎片

2.2标记整理

标记整理两个阶段

①把没有引用的对象标记为垃圾

②把可用的对象向前移动,让内存更加紧凑。连续的空间变多。

优点

没有内存碎片

缺点

对象整理过程中需要移动,垃圾回收效率低。

2.3复制

复制两个阶段

①把没有引用的对象标记为垃圾

②把活着的内存分为两块from区和to区。把from区存活的对象复制到to区。然后清空from区,然后交换from和to的位置

优点

没有内存碎片

缺点

占用双倍内存空间

3.分代垃圾回收

介绍

垃圾回收时,JVM结合3种算法协程工作。通过分代的垃圾回收机制,把堆内存划分为2块新生代和老年代,新生代(对象用完丢弃)划分为3个区域:伊甸园,幸存区From,幸存区To。

老年代(对象长时间使用) 不同的区域,算法不同。

 

工作机制

①创建新的对象占用伊甸园的内存空间,当伊甸园内存满时,触发垃圾回收Minor GC。沿着GC Roots引用链去找,如果对象没有被引用,就进行垃圾标记。然后采用复制的算法把伊甸园存活的对象复制到幸存区To中,并且对象寿命加1。然后交换from和to的位置。

 ②当伊甸园内存空间再次溢出时,触发第二次垃圾回收Minor GC。沿着GC Roots根引用链进行扫描伊甸园幸存区。把存活的对象复制到幸存区To中,并且对象寿命加1。然后交换from和to的位置。

 ③幸存区的对象寿命超过了阈值15,对象利用率高。晋升到老年代。(对象长时间使用

 ④当新生代老年代内存要溢出时,触发Full GC垃圾回收。

总结

①对象首先分配在伊甸园区域

②新生代空间不足时,触发Minor gc。伊甸园和from存活的对象,使用复制算法copy到to中。存活的对象年龄加1.交换from和to。

③minor gc会引发stop the world暂停其他用户的线程。垃圾回收结束,用户恢复运行。

④当对象寿命超过阈值时,会晋升到老年代,最大寿命是15

⑤老年代空间不足,触发Minor gc内存仍不足,那么full gc.

相关JVM参数

4.垃圾回收器

4.1串行垃圾回收器

介绍

①单线程垃圾回收器

②堆内存较小,适合个人电脑

③开启串行垃圾回收器 -XX:+UseSerialGC = Serial + SerialOld

Serial工作新生代,复制算法

SerialOld 工作在老年代,标记整理算法

工作流程

当堆内存要溢出时,触发垃圾回收。让其他线程在安全点停止下来,等待垃圾回收线程的结束。

 

4.2吞吐量优先:并行

介绍

①多线程

②堆内存较大,多核cpu支持

③让单位时间内,STW时间最短

工作流程

多核CPU中,有4个线程运行。当堆内存要溢出时,触发了垃圾回收。用户线程在安全点停止。此时的垃圾回收器会开启多个线程(跟CPU核数相关)进行垃圾回收。然后恢复其他线程的运行。

相关参数

-XX:+UseParallelGC ~ -XX:+UseParallelOldGC  开启吞吐量优先垃圾回收器

-XX:+UseAdaptiveSizePolicy  自适应大小策略,新生代

-XX:GCTimeRatio=ratio  调整吞吐量

-XX:MaxGCPauseMillis=ms 最大暂停毫秒数

-XX:ParallelGCThreads=n 运行时线程数

4.3响应时间优先

介绍

①多线程

②堆内存交大,多核cpu支持

③尽可能让单次STW时间变短

相关参数

-XX:+UseConcMarkSweepGC~-XX:+UseParNewGC~SerialOld 开启响应时间优先回收器

-XX:ParallelGCThreads=n~-XX:ConcGCThreads=threads 线程数

-XX:CMSInitiatingOccupancyFraction=percent 何时进行垃圾回收

-XX:+CMSScavengeBeforeRemark 新生代垃圾回收

工作流程

老年代发生内存泄漏,其他线程到达安全点暂停下来,垃圾回收器执行初始标记,用户线程恢复运行。垃圾回收线程进行并发标记,然后进行重新标记工作。垃圾回收线程进行并发清理。

5.G1垃圾回收器

简介

Garbage First

JDK 9默认

适用场景

①同时注重吞吐量和低延迟,默认暂停目标是200ms

②超大堆内存,会将堆划分多个大小相等的Region

③整体上是标记+整理算法,两个区域之间是复制算法。

相关JVM参数

-XX:+UseG1GC 开启G1垃圾回收

-XX:G1HeapRegisonSize=size 设置划分区域的大小

-XX:MaxGCPauseMills=time JVM最大暂停时间的目标值

5.1 G1垃圾回收阶段

1.Young Collection 新生代垃圾收集

①、G1会把堆内存划分成多个相等的区域。每个区域独立作为伊甸园,幸存区,老年代

②、新创建的对象会存储在伊甸园,当伊甸园内存满时会触发新生代垃圾回收机制,会STW。

③、伊甸园存活的对象使用copy算法到幸存区。

④、当幸存区的内存溢出时,触发新生代垃圾回收。幸存区对象存活年龄超过一定时间会晋升到老年代

2. Young Collection+Concurrent Mark新生代垃圾收集+并发标记

①在Young GC时进行GC Root的初始标记

②老年代占用堆空间比例达到阈值时,进行并发标记(不会STW)

-XX:InitiatingHeapOccupancyPercent=percent(默认45%)

3.Mixed Collection混合收集

介绍

会对E、S、O进行全面垃圾回收

①最终标记会STW

②拷贝存活会STW

-XX:MaxGCPauseMillis=ms

E:伊甸园区的存活对象通过复制算法到幸存区中

S:幸存区存活的对象复制到另一个幸存区,符合晋升条件的到老年区

O:老年代区把存活对象复制到另一个老年代区

优先收集垃圾最多的区,目的是达到暂停时间段的目标

5.2 Full GC     

1.SerialGC

①新生代内存不足发生的垃圾收集—minor gc

②老年代内存不足发生的垃圾收集—full gc

2.ParallelGC

①新生代内存不足发生的垃圾收集—minor gc

②老年代内存不足发生的垃圾收集—full gc

3.CMS

①新生代内存不足发生的垃圾收集—minor gc

②老年代内存不足,并发失败后才会full gc。

4.G1

①新生代内存不足发生的垃圾收集—minor gc

②老年代内存不足,老年代内存跟堆内存占比达到45%,触发并发标记和混合收集阶段。回收速度>垃圾产生速度,并发垃圾收集阶段。回收速度<垃圾产生速度,此时是full gc

5.3Young Collection跨代引用

1.新生代回收的跨代引用(老年代引用新生代)

①根对象有部分是来自老年代,老年代存活对象比较多,遍历效率低。所以采用CardTable把老年代区域进行细分成一个个的card。每个card是512k。如果老年代的对象引用了新生代的对象,就会把这个card标记为脏卡。只关注脏卡,减少搜索范围。

 ②脏卡引用了新生代的对象。新生代通过Remembered Set知道对应的脏卡。通过脏卡区域遍历GC Roots

5.4 Remark-重新标记

并发标记阶段的对象处理状态

 

黑色已经处理完成,存活的对象。灰色是处理当中的。白色是尚未处理的垃圾。

在并发标记阶段可能出现B对C引用,接着断开后C成为垃圾,然后A对C引用。此时C被判成垃圾了。

这个时候需要用到remark

Remark:对对象进行进一步检查,当对象引用发生改变时,会给引用提供一个写屏障(将C加入队列),接着进入重新标记阶段,对队列对象进行检查,有强引用的不标记为垃圾。

5.5 JDK8 u20字符串去重

指令

-XX:+UseStringDeduplication

规则

①将所有新分配的字符串放入一个队列

②当新生代回收时,G1并发检查是否有字符串重复

③如果值一样,让他们引用同一个char[]

优缺点

优点:节约大量内存

缺点:多占用CPU时间,新时代回收时间略微增加

跟String.intern()不一样

①String.intern()关注的是字符串对象

②字符串去重关注的是char[]

③JVM内部使用不同的字符串表

5.6 JDK8 u40并发标记类卸载

所有对象都经过并发标记后,知道哪些类不再使用。当一个类加载器所有类不再使用,则卸载它所加载的所有类

-xx:+ClassUnloadingWithConcurrentMark默认启用

5.7 JDK8 u60 回收巨型对象

①一个对象大于region的一半时,成为巨型对象。

②G1不会对巨型对象进行拷贝

③回收时优先考虑

④G1会跟踪老年代所有incoming引用。这样老年代incoming引用为0的巨型对象在新生代垃圾回收处理掉。

5.8 JDK9 并发标记起始时间的调整

①并发标记必须在堆空间占满前完成,否则退化为FullGC

②JDK 9之前需要使用 -XX:InitiatingHeapOccupancyPercent

③JDK 9可以调整:

-XX:InitiatingHeapOccupancyPercent 用来设置初始值

进行数据采样并动态调整

总会添加一个安全的空挡空间

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

9.4 数据库 TCP

#include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);//判断数据库对象是否包含了自己使用的数据库if(!db.contains("Stu.db")){//不存在数据库&#xff0…

在VScode中使用sftp传输本地文件到服务器端

安装SFTP 在VScode的扩展中安装sftp 注意这里需要在你没连接服务器的状态下安装&#xff0c;即本机需要有sftp 配置传输端口 安装成功后&#xff0c;使用快捷键"ctrlshiftp",输入sftp&#xff0c;选择Config 根据自己的实际情况修改配置文件&#xff0c;主要改h…

Python学习 -- datetime模块

当涉及到处理日期和时间数据时&#xff0c;Python的datetime模块提供了一系列类来帮助您执行各种操作。以下是各个类及其常用方法的详细介绍&#xff1a; date 类 date 类表示一个年、月、日的日期对象。以下是一些常用的 date 类方法&#xff1a; date.today()获取当前日期…

法国新法案强迫 Firefox 等浏览器审查网站

导读Mozilla 基金会已发起了一份请愿书&#xff0c;旨在阻止法国政府强迫 Mozilla Firefox 等浏览器审查网站。 据悉&#xff0c;法国政府正在制定一项旨在打击网络欺诈的 SREN 法案 (“Projet de loi Visant scuriser et reguler lespace numrique”)&#xff0c;包含大约 2…

ElasticSearch第三讲:ES详解 - Elastic Stack生态和场景方案

ElasticSearch第三讲&#xff1a;ES详解 - Elastic Stack生态和场景方案 本文是ElasticSearch第三讲&#xff0c;在了解ElaticSearch之后&#xff0c;我们还要了解Elastic背后的生态 即我们常说的ELK&#xff1b;与此同时&#xff0c;还会给你展示ElasticSearch的案例场景&…

<OpenCV> Mat属性

OpenCV的图像数据类型可参考之前的博客&#xff1a;https://blog.csdn.net/thisiszdy/article/details/120238017 OpenCV-Mat类型的部分属性如下&#xff1a; size&#xff1a;矩阵的大小&#xff0c; s i z e ( c o l s , r o w s ) size(cols,rows) size(cols,rows)&#xf…

(STM32H5系列)STM32H573RIT6、STM32H573RIV6、STM32H573ZIT6嵌入式微控制器基于Cortex®-M33内核

一、应用 工业&#xff08;PLC、工业电机控制、泵和压缩机&#xff09; 智能家居&#xff08;空调、冰箱、冰柜、中央警报系统、洗衣机&#xff09; 个人电子产品&#xff08;键盘、智能手机、物联网标签、跟踪设备&#xff09; 智能城市&#xff08;工业通信、照明控制、数字…

[react基础]关于v6版本route的变化,以及常见应用模式

该说不说,在做这些之前,你要记得一件事 route不是react或者vue等原本就有的组件!需要你手动下载!然后导入! 原本的框架只是最终挂载到一个html界面上!!! 别再问我为啥你扒下来的代码为啥不好使了! 讨厌 > _< 下载指令去看我另一篇推文 今天复盘了一下自己的实训…

区间DP 计数类DP 数位统计DP 状态压缩DP 树形DP 记忆化搜索

目录 区间DP石子合并分析思路代码实现 计数类DP整数划分完全背包DP的解法二维数组实现一维优化实现 另类DP状态表示的解法&#xff08;分拆数&#xff09;二维数组实现一维优化实现 数位统计DP计数问题注意代码实现 状态压缩DP蒙德里安的梦想实现思路朴素实现预处理优化实现 最…

L1 项目概述与Hadoop部署

1.技术栈&#xff1a;HadoopHiveSqoopFlumeAzkaban Flume采集Nginx web服务器上的日志&#xff0c;采集完成后存储到Hadoop的平台&#xff0c;最终存储到HDFS上&#xff0c;处理和分析采用Hive的方式&#xff0c;处理完之后利用Sqoop导出到Mysql中&#xff0c;最终利用一个Java…

Android逆向学习(一)vscode进行android逆向修改并重新打包

Android逆向学习&#xff08;一&#xff09;vscode进行android逆向修改并重新打包 写在前面 其实我不知道这个文章能不能写下去&#xff0c;其实我已经开了很多坑但是都没填上&#xff0c;现在专利也发出去了&#xff0c;就开始填坑了&#xff0c;本坑的主要内容是关于androi…

5、Nginx 配置实例-负载均衡

文章目录 5、Nginx 配置实例-负载均衡5.1 实现效果5.2 准备工作5.3 实验代码5.3.1、轮询&#xff08;默认&#xff09;5.3.2、weight5.3.3、ip_hash5.3.4、fair&#xff08;第三方&#xff09; 【尚硅谷】尚硅谷Nginx教程由浅入深 志不强者智不达&#xff1b;言不信者行不果。 …

docker常用中间件安装

文章目录 1、前言2、中间件安装2.1、mysql2.2、gitlab容器2.3、nacos2.4、redis2.5、xxljob2.6、zipkin2.7、sentinel2.8、seata2.8.1、获取镜像2.8.2、运行容器并获取配置 2.9、rockerMQ2.9.1、rockerMQ-namesrv2.9.2、rockerMQ-broker2.9.3、rockerMQ-console 2.10、jenkins2…

合宙Air724UG LuatOS-Air LVGL API控件-页面 (Page)

页面 (Page) 当控件内容过多&#xff0c;无法在屏幕内完整显示时&#xff0c;可让其在 页面 内显示。 示例代码 page lvgl.page_create(lvgl.scr_act(), nil) lvgl.obj_set_size(page, 150, 200) lvgl.obj_align(page, nil, lvgl.ALIGN_CENTER, 0, 0)label lvgl.label_crea…

机器学习算法详解1:基础知识合集

机器学习算法详解1&#xff1a;基础知识合集 前言 ​ 本系列主要对机器学习上算法的原理进行解读&#xff0c;给大家分享一下我的观点和总结。 本篇前言 ​ 开一个新系列&#xff0c;另外现在开学了&#xff0c;忙起来了&#xff0c;所以更新会很慢。 目录结构 文章目录 机器学…

彻底掌握Protobuf编码原理与实战

目录 1.类型2.VARINT 2.1 无符号数2.2 有符号数3.定长 3.1 I64类型3.2 I32类型4.LEN5.代码 学习这些有什么用&#xff1f; - 如果你是后端开发者&#xff0c;掌握这个对工作非常有用 - 如果你是求职者&#xff0c;面试时可以临危不惧 1.类型 最近看到有直接操作wire type相关的…

3D点云处理:基于角度的点云边缘点排序(附源码)

文章目录 0. 测试效果1. 基本内容2. 实现步骤3. 代码实现文章目录:3D视觉个人学习目录0. 测试效果 边缘点按照排序顺序显示(为便于显示查看,每隔五个点显示一个点) 1. 基本内容 基于角度的边缘点排序主要是基于每一个边缘点与点云中心位姿构成的向量与参考方向之间的…

deepin V23通过flathub安装steam畅玩游戏

deepin V23缺少32位库&#xff0c;在星火商店安装的steam,打开报错&#xff0c;无法使用&#xff01; 通过flathub网站安装steam,可以正常使用&#xff0c;详细教程如下&#xff1a; flathub网址&#xff1a;主页 | Flathub 注意&#xff1a;flathub下载速度慢&#xff0c;只…

vite+vue 项目使用 electron

创建 vitevue 项目 npm create viteElectron 官方文档 electron 安装 安装 electron npm install --save-dev electron新建 electron 的入口文件&#xff0c;我这里在根目录新建 electron 文件夹&#xff0c;然后新建main.js和preload.js文件 根据官网说明&#xff0c;将以下…

node版本问题

服务器下载下来的vue项目启动出现下列问题 npm ERR! path E:\vueEnv\app\node_modules\node-sass npm ERR! command failed npm ERR! command C:\Windows\system32\cmd.exe /d /s /c node scripts/build.js npm ERR! Building: C:\Program Files\nodejs\node.exe E:\vueEnv\ap…