学习笔记6——垃圾回收

学习笔记系列开头惯例发布一些寻亲消息

链接:https://baobeihuijia.com/bbhj/contents/3/190801.html
在这里插入图片描述

java垃圾回收(stop the world)

  • 专注于堆和方法区的垃圾回收,年轻代,老年代,永久代
  • 判断对象是否还存活?(区分尸体)
    • 引用计数算法
      • 给每个对象保存一个整型的引用计数器类型,用于记录对象被引用的情况
      • 缺点就是无法处理循环引用的情况,导致内存泄漏
    • 可达性分析算法
      • 找到GC ROOT对象,从根节点开始扩展访问
  • finalization机制(死前挣扎机制)
    • 用于在对象被回收时进行资源释放,比如关闭文件,套接字和数据库连接等。
    • 由于finalize的存在,对象有三种状态
      • 可触及的:活着
      • 可复活的:finalize重写过并且还没有执行,因此有机会进行死亡逃脱
      • 不可触及的:直接打死

在这里插入图片描述

  • 垃圾回收(三种算法清理尸体)

    • 复制算法(常用在新生代这种回收率较高的块上):有一块是专门用来备用的,清除的时候将存活对象复制到备用上,然后清除当前使用的内存块,交换两个的块的角色
    • 标记-清除算法:挨个清理,容易产生内存碎片,维护一个空闲列表,是一种非移动式的回收算法
    • 标记-压缩清除:在标记清除的基础上加上了内存碎片的整理,而不是维护一个列表,因此是一种移动式的回收算法
  • 增量收集算法:(解决stop the world)

    每次垃圾收集线程只收集一小片区域的内存空间,接着切换到应用程序线程。依次反复,直到垃圾收集完成。使用这种方式,由于在垃圾回收过程中,间断性地还执行了应用程序代码,所以能减少系统的停顿时间。但是,因为线程切换和上下文转换的消耗,会使得垃圾回收的总体成本上升,造成系统吞吐量的下降。

  • 分代算法和分区算法:针对不同情况用不同的垃圾回收算法

在这里插入图片描述

  • 内存溢出OOM和内存泄漏ML

  • stop the world,所有的GC都会有这种情况

  • GC的并发和并行

    • concurrent

      • 并发:多个事情在同一段时间内同时发生:用户线程和垃圾回收线程同时执行,不会停顿用户进程
    • STW

      • 并行:多个事情在用一个时间点上同时发生:多条垃圾回收并行,用户处于等待状态
      • 串行:相对于并行,单线程运行
  • 安全点: 数量需要适中,安全点实质上是一条指令(一般选择执行时间较长的指令作为安全点)

    • 抢先式中断:控制权在安全点,把所有线程挂起,没有运行到安全点的就恢复让运行到安全点
    • 主动式中断:线程运行到这里主动询问,主动挂起
  • 安全区:是扩展的安全点,在区域的任何位置GC都是安全的,安全区的对象引用关系不会变化

    • JVM会忽略标记进入到安全区的线程,线程还可以自主运行没有被停(因为其中的每句都是安全的)
    • 但等到线程出来的时候,就检查GC是否完成,完成了就可以走,没完成就得等着(要是进入了不安全区域,而且这个线程还没有被停住,就容易让JVM和改变后的引用关系发生冲突)
  • 引用(是有引用的可触及的对象,所以不会被判定为尸体,但是不一定有引用就会存活,也分情况)

    • 强引用(有用)

      • 只要引用关系存在,JVM宁愿OOM也不会回收对象
      • 是造成内存泄漏的原因之一
      • 强引用可以直接访问目标对象
    • 软引用(有用但是非必须)

      • 即将内存溢出时,回收这些软可达对象,回收后内存还是不够才抛出异常
    • weak弱引用(非必须)

      • 不管堆空间使用情况,下一次垃圾回收器工作一定会回收掉
      • GC的优先级很低,一般很久才能拿到线程,所以也能保有很久
      • 当系统内存不足时,这些缓存数据会被回收,不会导致内存溢出。而当内存资源充足时,这些缓存数据又可以存在相当长的时间,从而起到加速系统的作用。软引用、弱引用都非常适合来保存那些可有可无的缓存数据。
    • 虚引用:虚引用不会对对象的生存造成影响(如果只有虚引用,那么和没有引用是一样的)

      • 唯一作用就是对象回收时会收到一个系统通知,可以用来跟踪对象回收时间
    • 终结器引用(并不是一种对象还被别的对象需要的信号,而是将所有需要执行finalize的对象全都加进来

      • Finalizer线程通过终结器引用找到对象,并执行finalize方法
  • 垃圾回收器

    • 评价指标:吞吐量、暂停时间、占用内存

      • 个人理解:吞吐量是指用户的实际有效运行时间占比,暂停时间是用户需要等待GC的时间,这两者是相互冲突的。因为想要有效占比时间长,那么就需要GC不要那么频繁的发起回收,而是一次性高效回收完,回收时间较长,而暂停时间短,就需要每次GC的时间很短,但是发起次数比较频繁,导致程序吞吐量下降。
    • 不同垃圾回收器(年轻代和老年代的垃圾回收器进行组合)

      • 新生代:
        • serial:复制算法、串行回收、stop the world
        • ParNew:复制算法、并行回收、stop the world
        • parallel scavenge:复制算法、并行回收、stop the world、可控吞吐量(适用于不需要太多交互的场景)

      在这里插入图片描述

    在这里插入图片描述

    • 老年代

      • serial old:标记-压缩算法、串行回收、stop the world
      • parallel old:标记-压缩算法,并行,stop the world
      • CMS: 标记-清除算法、并发、stop the world,必须采用空闲列表而不是指针碰撞(与parallel scavenge不符,只能换别的新生代回收器)由于CMS回收过程中,应该确保应用程序也有部分的内存,所以CMS不能等到老年代全部满了再进行收集,到一定阈值就应该收集。当预留内存不满足应用程序要求的时候,就会出现concurrent failure,虚拟机会临时启动serial old回收,停顿时间变长。
        • 初始标记STW:GC Root
        • 并发标记Concurrent:遍历对象图
        • 重新标记STW:修正并发标记期间的错误
        • 并发清除Concurrent:清除标记阶段死亡的对象,释放内存空间

      外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
      在这里插入图片描述

    • G1回收器:全能垃圾回收器

      • 将堆内存分为不同的region,尽量避免对整个堆全区域进行垃圾回收。
      • 跟踪每个区域的垃圾回收价值,维护一个优先列表,根据每次允许的收集时间优先回收价值最大的region
      • 并行与并发兼容:部分GC并行进行,部分工作可以与应用程序并发交替执行

      在这里插入图片描述

    • 流程

      • 年轻代回收过程:G1独占式并行运行,首先收集年轻代eden和survivor的内存回收片段,从年轻代移动存活对象到survivor或老年区间(最频繁使用的)

        • Rset避免全堆扫描root
        • 更新Rset(有的被老年代引用了)
        • 复制:将存活片段复制到survivor,空间不够就直接到old
      • 老年代并发标记过程:堆内存使用达到阈值(隔几天进行一次)

        • 初始标记阶段STW:标记根节点直接可达对象,触发Young GC
        • 根区域扫描(检查survivor直接可达老年代对象并标记)
        • 并发标记:计算区域的对象活性(存活比例)
        • 再次标记:修正并发标记的结果
        • 独占清理STW:计算存活对象和GC回收比例进行排序
        • 并发清理阶段
      • 老年代的混合回收:每次只扫描一小部分老年代的region,然后老年代和年轻代一起回收,为了避免堆内存被耗尽

      • Full GC:会产生大的停顿,尽量避免

    • Remember set

      在这里插入图片描述

      在这里插入图片描述

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

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

相关文章

Java8Stream快速使用

将List集合存入流中 List<String> list new ArrayList<>();list.add("张一");list.add("张二");list.add("张三");list.add("李四");list.add("赵五");list.add("张六");list.add("王八"…

Linux进程通信——IPC、管道、FIFO的引入

进程间的通信——IPC 进程间通信 (IPC&#xff0c;InterProcess Communication) 是指在不同进程之间传播或交换信息。 IPC的方式通常有管道 (包括无名管道和命名管道) 、消息队列、信号量、共享存储、Socket、Streams等。其中 Socket和Streams支持不同主机上的两个进程IPC。 …

pyQt主界面与子界面切换简易框架

本篇来介绍使用python中是Qt功能包&#xff0c;设置一个简易的多界面切换框架&#xff0c;实现主界面和多个子界面直接的切换显示。 1 主界面 设计的Demo主界面如下&#xff0c;主界面上有两个按钮图标&#xff0c;点击即可切换到对应的功能界面中&#xff0c;进入子界面后&a…

【mediasoup】TransportCongestionControlClient 1: 代码走读

TransportCongestionControlClient 基于m77版本的libwebrtc ,但是TransportCongestionControlClient 并不是libwebrt中的,是mediasoup自己封装实现:TransportCongestionControlClient 用于发送端D:\XTRANS\soup\mediasoup-sfu-cpp\worker\src\RTC\TransportCongestionContro…

(动手学习深度学习)第13章 实战kaggle竞赛:CIFAR-10

导入相关库 import collections import math import os import shutil import pandas as pd import torch import torchvision from torch import nn from d2l import torch as d2l下载数据集 d2l.DATA_HUB[cifar10_tiny] (d2l.DATA_URL kaggle_cifar10_tiny.zip,2068874e4…

ElasticSearch在Windows上的下载与安装

Elasticsearch是一个开源的分布式搜索和分析引擎&#xff0c;它可以帮助我们快速地搜索、分析和处理大量数据。Elasticsearch能够快速地处理结构化和非结构化数据&#xff0c;支持全文检索、地理位置搜索、自动补全、聚合分析等功能&#xff0c;能够承载各种类型的应用&#xf…

基于Qt中操作MySQL数据库示例

# 一、安装驱动 ##(1)安装 在Qt中操作MySQL数据库首先要安装mysql的驱动文件,将MySQL下的libmusql.dll文件复制到Qt的安装路径下的bin文件夹下即可。 直接将libmysql.dll文件粘贴到此文件夹中。 ## (2)验证驱动是否安装成功 复制成功之后来测试一下驱动程序是否安装成功…

◢Django 分页+搜索

1、搜索数据 从数据库中获取数据&#xff0c;并进行筛选&#xff0c;xx__contains q作为条件&#xff0c;查找的是xx列中有q的所有数据条 当有多个筛选条件时&#xff0c;将条件变成一个字典&#xff0c;传入 **字典 &#xff0c;ORM会自行翻译并查找。 筛选电话号码这一列…

Java获取指定日期到当前日期的差距

Java获取指定日期到当前日期的差距 一、指定日期到今天的y年m月d日 private JSONObject getYesrMonthDay(String dataParam){JSONObject res new JSONObject();/*只比较年月日&#xff0c;不要时间*/DateTimeFormatter df DateTimeFormatter.ofPattern("yyyy-MM-dd&quo…

在Go编程中调用外部命令的几种场景

1.摘要 在很多场合, 使用Go语言需要调用外部命令来完成一些特定的任务, 例如: 使用Go语言调用Linux命令来获取执行的结果,又或者调用第三方程序执行来完成额外的任务。在go的标准库中, 专门提供了os/exec包来对调用外部程序提供支持, 本文将对调用外部命令的几种使用方法进行总…

【机器学习】034_多层感知机Part.2_从零实现多层感知机

一、解决XOR问题 1. 回顾XOR问题&#xff1a; 如图&#xff0c;如何对XOR面进行分割以划分四个输入 对应的输出 呢&#xff1f; 思路&#xff1a;采用两个分类器分类&#xff0c;每次分出两个输入 &#xff0c;再借助这两个分类从而分出 。 即采用同或运算&#xff0c;当两…

通过easyexcel实现数据导入功能

上一篇文章通过easyexcel导出数据到excel表格已经实现了简单的数据导出功能&#xff0c;这篇文章也介绍一下怎么通过easyexcel从excel表格中导入数据。 目录 一、前端代码 index.html index.js 二、后端代码 controller service SongServiceImpl 三、功能预览 四、后端…

WordPress画廊插件Envira Gallery v1.9.7河蟹版下载

Envira Gallery是一款功能强大的WordPress画廊插件。通过使用这个插件&#xff0c;你可以在WordPress的前台页面上创建出令人赏心悦目的图片画廊展示形式。 拖放生成器&#xff1a;轻松创建精美照片和视频画廊 自定义主题&#xff0c;打造独特外观 使用预设模板&#xff0c;为…

分类预测 | Matlab实现基于PSO-SDAE粒子群优化算法优化堆叠去噪自编码器的数据分类预测

分类预测 | Matlab实现基于PSO-SDAE粒子群优化算法优化堆叠去噪自编码器的数据分类预测 目录 分类预测 | Matlab实现基于PSO-SDAE粒子群优化算法优化堆叠去噪自编码器的数据分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现基于PSO-SDAE粒子群优化算法…

使用树莓派学习Linux系统编程的 --- 库编程(面试重点)

在之前的Linux系统编程中&#xff0c;学习了文件的打开&#xff1b;关闭&#xff1b;读写&#xff1b;进程&#xff1b;线程等概念.... 本节补充“Linux库概念 & 相关编程”&#xff0c;这是一个面试的重点&#xff01; 分文件编程 在之前的学习中&#xff0c;面对较大的…

算法---相等行列对

题目 给你一个下标从 0 开始、大小为 n x n 的整数矩阵 grid &#xff0c;返回满足 Ri 行和 Cj 列相等的行列对 (Ri, Cj) 的数目。 如果行和列以相同的顺序包含相同的元素&#xff08;即相等的数组&#xff09;&#xff0c;则认为二者是相等的。 示例 1&#xff1a; 输入&…

原理Redis-Dict字典

Dict 1) Dict组成2) Dict的扩容3) Dict的收缩4) Dict的rehash5) 总结 1) Dict组成 Redis是一个键值型&#xff08;Key-Value Pair&#xff09;的数据库&#xff0c;可以根据键实现快速的增删改查。而键与值的映射关系正是通过Dict来实现的。 Dict由三部分组成&#xff0c;分别…

无重复最长字符串(最长无重复子字符串),剑指offer,力扣

目录 原题&#xff1a; 力扣地址&#xff1a; 我们直接看题解吧&#xff1a; 解题方法&#xff1a; 难度分析&#xff1a; 难度算中下吧&#xff0c;这个总体不算很难&#xff0c;而且滑动窗口&#xff0c;以及哈希都比较常见 审题目事例提示&#xff1a; 解题思路&#xff08;…

vue3 setup展示数据

效果图 1.创建数据 content.js import { reactive } from vueconst data reactive({color:red,title: 二十四节气,subTitle: 节气&#xff0c;是干支历中表示自然节律变化以及确立“十二月建”&#xff08;月令&#xff09;的特定节令。,list: [{name: "立春",con…

代码随想录-刷题第二天

977. 有序数组的平方 题目链接&#xff1a;977. 有序数组的平方 思路&#xff1a;双指针思想&#xff0c;数组是有序的且含有负数&#xff0c;其中元素的平方一定是两边最大。定义两个指针&#xff0c;从两端开始向中间靠近&#xff0c;每次比较两个指针的元素平方大小&#…