25.02.04 《CLR via C#》 笔记14

第二十一章 托管堆和垃圾回收

  1. 内存分配过程 CLR维护一个“下一次分配指针”(NextObjPtr),指向当前托管堆中第一个可用的内存地址
    1. 计算类型所需的字节数,加上对象开销(类型对象指针、同步块索引)所需字节数,检查空间是否足够
    2. 如果空间足够,更新指针到分配的对象末尾
    3. 如果空间不足,触发垃圾回收释放未使用内存
  2. 垃圾回收算法
    1. 把引用类型变量称为根
    2. 标记:暂停所有线程,遍历堆中所有对象,在同步块索引上标记为应该删除(不可达);如果任何根引用了堆上的对象,则标记为可达
    3. 压缩:移动幸存的对象,使它们占用连续的空间(类似碎片整理),同时更新被移动对象的地址
    4. 移动NextObjPtr,指向最后一个幸存对象之后的位置
    5. 如果回收不了内存,说明内存已耗尽,抛出异常
  3. 静态字段引用的对象一直存在,可能会使其引用的对象一直存活,造成内存泄漏
  4. 分代回收算法(原理:新分配的对象生存周期短,老对象生存周期长)
    1. 新创建的对象称为第0代对象,CLR初始化时为第0代选择一个预算容量,当容量不足时触发垃圾回收,幸存的对象就是第1代
    2. 如果根或对象引用了老一代的某个对象,那么可以忽略老对象内部的所有引用;如果老对象引用了新对象,设置了其他机制标记
    3. 如果第1代也用完预算,则变为第2代,同时第0代变为第1代;最高就是第2代,没有第3代
    4. 预算大小是CLR根据每次回收垃圾的多少来动态决定的(启发式算法)
    5. 如果第0代所有对象都是垃圾,可以通过将NextObjPtr移到起始处直接完成垃圾回收
  5. 垃圾回收触发条件
    1. 代码显式调用Collect方法回收(不推荐)
    2. windows报告内存低
    3. CLR卸载AppDomain 执行所有代的回收
    4. CLR关闭 进程终止,由windows回收进程全部内存
  6. 大对象 大于85KB的大对象专门分配在大对象堆,GC不会压缩大对象,大对象总是第2代
  7. 垃圾回收模式
    1. 工作站模式:针对单用户环境优化,单线程垃圾回收,低延迟优先
    2. 服务器模式:针对多用户优化,多线程垃圾回收,堆空间更大,暂停时间较长
  8. 应尽量避免使用Finalize:这个方法在GC结束后调用,由于该方法中可能使用到对象的字段,导致对象在回收时延长了存活时间;Finalize方法的执行时间和顺序是不可控制的;Finalize方法在专用的线程上执行,一旦阻塞则无法调用其他Finalize方法造成内存一直泄露
  9. 封装本机资源时推荐从SafeHandle类派生,以保证本机资源在垃圾回收时释放
  10. IDisposable接口
    1. 如果类定义的一个字段实现了dispose模式,那么类本身也应该实现dispose模式
    2. 对象被显式dispose后,对象本身依然存在
    3. using语句初始化对象,编译器会自动生成try-finally块,并在finally中调用Dispose方法
  11. StreamWriter包装了一个Stream(FileStream或MemoryStream),如果没有显式调用Dispose,会造成缓冲区丢失(StreamWriter不支持终结,就算支持,如果Stream先终结,StreamWriter也无法正确终结)
  12. 本机资源可能占用大量内存,而占用很小的托管内存,为使GC能正确感知内存压力,及时执行垃圾回收,可以使用AddMemoryPressure、RemoveMemoryPressure来修正
  13. Finalize的原理:维护一个终结列表和F-Reachable队列,当对象的实例构造器被调用前,如果定义了Finalize方法,则加入终结列表;被当可终结对象被回收时,离开终结列表进入freachable 队列(此时对象又有引用了(被这个队列引用),不再是垃圾),然后由专用线程调用队列中对象的Finalize方法;调用完成后,对象可能被提升到较老的一代,然后等待下次被回收(此时已经不在终结列表中了,这次会被正常回收)。
  14. GCHandle类提供了对托管对象的直接控制,避免垃圾回收器意外回收对象
    1. 标志位GCHanleType定义

      1. Weak,弱引用,此类型的句柄不会阻止垃圾回收器回收对象。如果对象被垃圾回收,句柄会指向 null,回收后不可访问,无法恢复
      2. WeakTrackResurrection,弱引用,但Finalize调用后仍然可达,在终结器完成之前句柄仍有效
      3. Normal,普通句柄,对象在使用该类型的句柄时不会被垃圾回收,必须手动释放句柄
      4. Pinned,固定句柄,对象固定在内存中,垃圾回收器不能移动它
    2. 弱引用:当对象只通过弱引用被引用时,垃圾回收器会认为该对象是不可达的,从而可以对其进行回收

    3. 用GCHandle创建一个对象的弱引用(Weak),当GCHandle.Target返回null时表示对象已经被回收

    4. 用GCHandle创建一个对象的Normal引用,然后将此对象传给非托管代码,非托管代码就可以使用GCHandle.Target来获取托管对象的指针,这能保证托管对象一直存活、在内存中移动也不丢失;如果已用完,需用Free方法释放

    5. 用GCHandle创建一个对象的Pinned引用,然后用GCHandle.AddrOfPinnedObject获得已固定对象的指针,保证该地址中的数据在内存中不会移动

    6. fixed语句可以直接在unsafe代码块中临时固定对象

    7. ConditionalWeakTable:实现了对Key是弱引用,但只要key未被回收,value一定未被回收

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

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

相关文章

解读“大语言模型(LLM)安全性测评基准”

1. 引入 OWASP,全称为Open Web Application Security Project,即开放式Web应用程序安全项目,是一个致力于提高软件安全性的非营利国际组织。 由于庞大的规模和复杂的结构,大语言模型也存在多种安全风险,如prompt误导…

【大数据技术】教程03:本机PyCharm远程连接虚拟机Python

本机PyCharm远程连接虚拟机Python 注意:本文需要使用PyCharm专业版。 pycharm-professional-2024.1.4VMware Workstation Pro 16CentOS-Stream-10-latest-x86_64-dvd1.iso写在前面 本文主要介绍如何使用本地PyCharm远程连接虚拟机,运行Python脚本,提高编程效率。 注意: …

Notepad++消除生成bak文件

设置(T) ⇒ 首选项... ⇒ 备份 ⇒ 勾选 "禁用" 勾选禁用 就不会再生成bak文件了 notepad怎么修改字符集编码格式为gbk 如图所示

CSS布局(一)flex一篇搞定

目录 一、flex布局 1.1. 认识flex布局 1.2. flex布局重要的概念 二、flex container中的属性 2.1.flex-direction 2.2.flex-wrap、flex-flow 2.3.justify-content 2.4.align-items 2.5.align-content 三、 flex item中的属性 3.1.order 3.2.align-self 3.3.flex-gr…

e2studio开发RA2E1(5)----GPIO输入检测

e2studio开发RA2E1.5--GPIO输入检测 概述视频教学样品申请硬件准备参考程序源码下载新建工程工程模板保存工程路径芯片配置工程模板选择时钟设置GPIO口配置按键口配置按键口&Led配置R_IOPORT_PortRead()函数原型R_IOPORT_PinRead()函数原型代码 概述 本篇文章主要介绍如何…

[吾爱出品]CursorWorkshop V6.33 专业鼠标光标制作工具-简体中文汉化绿色版

CursorWorkshop V6.33 专业鼠标光标制作工具 链接:https://pan.xunlei.com/s/VOIFeq5DFB9FS56Al_mT2EfdA1?pwd7ij4# 产品概述 Axialis CursorWorkshop 是一个专业光标创作工具它在 Windows 下运行,让您轻松创建高质量的静态和动态光标适用于 Windows …

STM32单片机学习记录(2.2)

一、STM32 13.1 - PWR简介 1. PWR(Power Control)电源控制 (1)PWR负责管理STM32内部的电源供电部分,可以实现可编程电压监测器和低功耗模式的功能; (2)可编程电压监测器(…

【物联网】ARM核常用指令(详解):数据传送、计算、位运算、比较、跳转、内存访问、CPSR/SPSR

文章目录 指令格式(重点)1. 立即数2. 寄存器位移 一、数据传送指令1. MOV指令2. MVN指令3. LDR指令 二、数据计算指令1. ADD指令1. SUB指令1. MUL指令 三、位运算指令1. AND指令2. ORR指令3. EOR指令4. BIC指令 四、比较指令五、跳转指令1. B/BL指令2. l…

Nacos 的介绍和使用

1. Nacos 的介绍和安装 与 Eureka 一样,Nacos 也提供服务注册和服务发现的功能,Nacos 还支持更多元数据的管理, 同时具备配置管理功能,功能更丰富。 1.1. windows 下的安装和启动方式 下载地址:Release 2.2.3 (May …

【零基础到精通】小白如何自学网络安全

小白人群想学网安但是不知道从哪入手?一篇文章告诉你如何在4个月内吃透网安课程,掌握网安技术 一、基础阶段 1.了解网安相关基础知识 了解中华人民共和国网络安全法、熟知网络安全的相关概念:包括信息安全、风险管理、网络攻防原理、认证与…

架构规划之任务边界划分过程中承接分配

架构师在边界划分的过程中需要做什么事情呢?接下来,我们会讨论一些关于任务分配的 基础假设,以及由这些基础假设而带来的决策路径。 所谓任务边界划分,就是判定某个任务在多个承接方中,应该归属到哪个承接方的过程。…

如可安装部署haproxy+keeyalived高可用集群

第一步,环境准备 服务 IP 描述 Keepalived vip Haproxy 负载均衡 主服务器 Rip:192..168.244.101 Vip:192.168.244.100 Keepalive主节点 Keepalive作为高可用 Haproxy作为4 或7层负载均衡 Keepalived vip Haproxy 负载均衡 备用服务…

MySQL常用数据类型和表的操作

文章目录 (一)常用数据类型1.数值类2.字符串类型3.二进制类型4.日期类型 (二)表的操作1查看指定库中所有表2.创建表3.查看表结构和查看表的创建语句4.修改表5.删除表 (三)总代码 (一)常用数据类型 1.数值类 BIT([M]) 大小:bit M表示每个数的位数,取值范围为1~64,若…

DeepSeekMoE:迈向混合专家语言模型的终极专业化

一、结论写在前面 论文提出了MoE语言模型的DeepSeekMoE架构,目的是实现终极的专家专业化(expert specialization)。通过细粒度的专家分割和共享专家隔离,DeepSeekMoE相比主流的MoE架构实现了显著更高的专家专业化和性能。从较小的2B参数规模开始&#x…

寻迹传感器模块使用说明

产品用途: 1、电度表脉冲数据采样 2、传真机碎纸机纸张检测 3、障碍检测 4、黑白线检测 产品介绍: 1、采用 TCRT5000 红外反射传感器 2、检测反射距离:1mm~25mm 适用 3、比较器输出,信号干净,波形好,驱…

java项目验证码登录

1.依赖 导入hutool工具包用于创建验证码 <dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.5.2</version></dependency> 2.测试 生成一个验证码图片&#xff08;生成的图片浏览器可…

Baklib探讨如何通过内容中台提升组织敏捷性与市场竞争力

内容概要 在数字化转型的浪潮中&#xff0c;内容中台已经成为企业提升市场响应速度和竞争力的关键所在。内容中台不仅是信息处理的集结地&#xff0c;更是促进资源高效整合和灵活应用的重要平台。通过构建一个高效的内容中台架构&#xff0c;企业能够更好地应对不断变化的市场…

Java基础——分层解耦——IOC和DI入门

目录 三层架构 Controller Service Dao ​编辑 调用过程 面向接口编程 分层解耦 耦合 内聚 软件设计原则 控制反转 依赖注入 Bean对象 如何将类产生的对象交给IOC容器管理&#xff1f; 容器怎样才能提供依赖的bean对象呢&#xff1f; 三层架构 Controller 控制…

Spring中@Conditional注解详解:条件装配的终极指南

一、为什么要用条件装配&#xff1f; 在实际开发中&#xff0c;我们经常需要根据不同的运行环境、配置参数或依赖情况动态决定是否注册某个Bean。例如&#xff1a; 开发环境使用内存数据库&#xff0c;生产环境连接真实数据库 当存在某个类时才启用特定功能 根据配置文件开关…

Redis代金卷(优惠卷)秒杀案例-多应用版

Redis代金卷(优惠卷)秒杀案例-单应用版-CSDN博客 上面这种方案,在多应用时候会出现问题,原因是你通过用户ID加锁 但是在多应用情况下,会出现两个应用的用户都有机会进去 让多个JVM使用同一把锁 这样就需要使用分布式锁 每个JVM都会有一个锁监视器,多个JVM就会有多个锁监视器…