【C#学习笔记】内存管理

news2025/1/21 1:01:15

在这里插入图片描述

文章目录

  • 分配内存
  • 释放内存
  • GC
    • 标记清除算法
    • 分代算法
  • .NET的GC机制有这样两个问题:


官方文档

自动内存管理

自动内存管理是CLR在托管执行过程中提供的服务之一。 公共语言运行时的垃圾回收器为应用程序管理内存的分配和释放。 对开发人员而言,这就意味着在开发托管应用程序时不必编写执行内存管理任务的代码。 自动内存管理可解决常见问题,例如,忘记释放对象并导致内存泄漏,或尝试访问已释放对象的内存。

分配内存

总的来说,C#内存是分为四个区块的,分别是

  • 全局数据区:存放全局变量,静态数据,常量
  • 代码区:存放所有程序代码
  • 栈区:存放为运行而分配的局部变量,参数,返回数据,返回地址等
  • 堆区:即自由存储区

在进程中,代码执行的两大区域,是栈和堆。在栈中,代码以栈的形式依次入栈出栈执行,每个线程会维护一个属于自己的线程栈,而栈区的内存地址从上往下增加:
摘自一文读懂C#的 堆、栈、值类型、引用类型
深入理解C#中的堆(Heap)与栈(Stack),一次性全都掌握!
C#垃圾回收机制详解
在这里插入图片描述

相信对于代码在栈中的执行,不少计算机专业的都相当清楚。而在堆中却不是这样,初始化新进程时,运行时会为进程保留一个连续的地址空间区域。 这个保留的地址空间被称为托管堆。

托管堆是专门用于C#的自动内存管理的,它由CLR进行管理。

托管堆维护着一个指针,用它指向将在堆中分配的下一个对象的地址。最初,该指针设置为指向托管堆的基址。与栈不同,堆是从下往上的,因此所有自由空间都在已用空间的上面。而与栈不同的最大特点就是,我们往堆中存取时是自由存取的,可以以任意顺序进行存取和移除。
在这里插入图片描述
(乱七八糟的堆)

记得我在之前曾经讲过,C#的变量类型存在两种,第一种值类型,它是直接存储在栈上的,且使用值类型赋值,则需要完全复制一个副本存在堆栈上。而我们的操作都是在栈上实现的,因为栈上的操作速度快,内存分配和释放简单。

而引用类型则方便许多,一方面,引用类型的本体是存储在堆中的,在堆中的存取更加自由。另一方面,我们会往栈中内存入栈存储了堆地址变量。当我们想要访问引用类型时,只需通过栈中的地址就可以找到它。但是却不能直接访问堆,因为数据存储太随意,如果没有栈中的指针,我们根本不知道它在堆中的什么位置。

在这里插入图片描述
但是使用堆引发一个问题:在栈中,由于程序执行完毕,各种入栈的变量会自动释放。但是堆首先存取随意,如果程序员忘了释放用完的变量,就会浪费内存。其次堆中的变量就算释放了,那么被释放的变量对应内存就空在那里了。如果空着的连续内存太小我们根本用不了,就会产生内存碎片。因此C#使用托管堆,由GC机制帮助我们自动管理这个堆。

应用程序创建第一个引用类型时,将为托管堆的基址中的类型分配内存。
应用程序创建下一个对象时,垃圾回收器在紧接第一个对象后面的地址空间内为它分配内存。
只要地址空间可用,垃圾回收器就会继续以这种方式为新对象分配空间。

从托管堆中分配内存要比非托管内存分配速度快。 由于运行时通过为指针添加值来为对象分配内存,所以这几乎和从堆栈中分配内存一样快。
另外,由于连续分配的新对象在托管堆中是连续存储,所以应用程序可以快速访问这些对象。

总结:栈主要用于存储局部变量和方法调用信息,它的操作速度较快;而堆主要用于存储动态分配的对象,它的操作速度较慢但能够支持更长的生命周期和对象共享。对于数据较小、生命周期短暂的变量,可以考虑使用栈;对于较大的对象或需要长时间存活的对象,需要使用堆。


释放内存

让我们了解一下GC机制是怎么释放内存的。

.NET 的垃圾回收器管理应用程序的内存分配和释放。 每当有对象新建时,公共语言运行时都会从托管堆为对象分配内存。 只要托管堆中有地址空间,运行时就会继续为新对象分配空间。 不过,内存并不是无限的。 垃圾回收器最终必须执行垃圾回收来释放一些内存。 垃圾回收器的优化引擎会根据所执行的分配来确定执行回收的最佳时机。 执行回收时,垃圾回收器会在托管堆中检查应用程序不再使用的对象,然后执行必要的操作来回收其内存。

当满足以下条件之一时将发生垃圾回收:

  • 系统具有低的物理内存。 内存大小是通过操作系统的内存不足通知或主机指示的内存不足检测出来的。

  • 由托管堆上已分配的对象使用的内存超出了可接受的阈值。 随着进程的运行,此阈值会不断地进行调整。

  • 调用 GC.Collect 方法。 几乎在所有情况下,你都不必调用此方法,因为垃圾回收器会持续运行。 此方法主要用于特殊情况和测试。

GC

总体来看C#的GC和Lua的GC其实差不多

引用跟踪:GC的第一步是通过引用跟踪,确定哪些对象仍然被活动的引用所引用。GC会从栈上的根对象开始,根对象包括方法中的局部变量和静态变量。然后,从根对象开始,跟踪引用链,找出所有的可达对象。

根对象标记:在引用跟踪的过程中,GC会将所有可达的对象标记为活动对象。这些活动对象将被保留下来,不会被清理。

标记清除算法

对象标记:GC接下来会遍历堆中的所有对象,从根对象可达的对象开始,将它们标记为活动对象。GC使用一种叫做“标记-清除(mark and sweep)”的算法来标记对象。首先所有的堆中的对象全部默认为垃圾,接着通过遍历栈找到引用的对象,给这些被引用的对象打上标记。最后将所有的垃圾内存释放(或者标记为空闲状态)。

压缩阶段:当清理完垃圾之后,堆中的内存会变得零零散散的,为了避免内存碎片,接着就会把所有存活的对象移动到堆的底部。

在这里插入图片描述

分代算法

分代算法的原理基于统计学,简单来说活动时间越长的对象使用率越高,则死亡率越低,越不需要进行清理。

分代算法的假设前提条件:
1、大量新创建的对象生命周期都比较短,而较老的对象生命周期会更长
2、对部分内存进行回收比基于全部内存的回收操作要快
3、新创建的对象之间关联程度通常较强。heap分配的对象是连续的,关联度较强有利于提高CPU cache的命中率

.NET将heap分成3个代龄区域: Gen 0、Gen 1、Gen 2
在这里插入图片描述
每次迭代时使用标记清除算法去除垃圾。而堆将被分为3个代龄区域,相应的GC有3种方式: # Gen 0 collections, # Gen 1 collections, #Gen 2 collections

当Gen0区域内存达到阈值将触发# Gen 0 collections后,存活的对象将进入Gen1区域;

当Gen1区域内存达到阈值将触发 # Gen 1 collections,将Gen0存活对象放入Gen1,Gen1存活对象放入Gen2。

当Gen2区域内存达到阈值将触发 # Gen 2 collections,将Gen0存活对象放入Gen1,Gen1存活对象放入Gen2,Gen2存活对象位置不变。

Gen 0和Gen 1比较小,这两个代龄加起来总是保持在16M左右;Gen2的大小由应用程序确定,可能达到几G,因此0代和1代GC的成本非常低,2代GC称为fullGC,通常成本很高。粗略的计算0代和1代GC应当能在几毫秒到几十毫秒之间完成,Gen 2 heap比较大时fullGC可能需要花费几秒时间。大致上来讲.NET应用运行期间2代、1代和0代GC的频率应当大致为1:10:100。

和Lua一样,当对象被GC清理的时候,会调用一个终结器函数。

.NET的GC机制有这样两个问题:

首先,GC并不是能释放所有的资源。它不能自动释放非托管资源。

第二,GC并不是实时性的,这将会造成系统性能上的瓶颈和不确定性。

GC并不是实时性的,这会造成系统性能上的瓶颈和不确定性。所以有了IDisposable接口,IDisposable接口定义了Dispose方法,这个方法用来供程序员显式调用以释放非托管资源。使用using 语句可以简化资源管理。

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

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

相关文章

Ubuntu 虚拟机和主机无法互相复制文字和文件

1.在虚拟机列表中,右键查看是否有安装VMware Tools,如果没有安装点击安装,如果已经安装了,上面显示重现安装VMware Tools,并且为灰色,如图: 2.如果没有安装点击安装,如果已经安装&am…

【数据结构】堆的原理实现

🔥🔥 欢迎来到小林的博客!!       🛰️博客主页:✈️林 子       🛰️博客专栏:✈️ 数据结构与算法       🛰️社区 :✈️ 进步学堂       &am…

C语言的浮点类型:float,double,long double

文章目录 浮点型概述浮点型常量浮点型常量的后缀 溢出上溢 overflow下溢 underflow 一个特殊的浮点值:NaN初始化 浮点型概述 float 称为单精度浮点型。 double 称为双精度浮点型。 long double 称为长双精度浮点型。 C 标准规定,float 必须至少精确到…

MySQL存储过程(二十四)

你相信吗, 相信那一天的夕阳吗? 上一章简单介绍了 MySQL的索引(二十三),如果没有看过,请观看上一章 一. 存储过程 MySQL从5.0版本开始支持存储过程和函数。存储过程和函数能够将复杂的SQL逻辑封装在一起, 应用程序无须关注存储过程和函数内部复杂的S…

【DBeaver】CLIENT_PLUGIN_AUTH is required

1、右键打开编辑连接弹窗,进入编辑驱动设置 2、选择“库” 3、全部删掉,然后点击“重置为默认状态” 4、选中然后点击“下载更新” 5、点击版本号,选择与mysql相同的版本 6、最后一步

使用可视化docker浏览器,轻松实现分布式web自动化

01、前言 顺着docker的发展,很多测试的同学也已经在测试工作上使用docker作为环境基础去进行一些自动化测试,这篇文章主要讲述我们在docker中使用浏览器进行自动化测试如果可以实现可视化,同时可以对浏览器进行相关的操作。 02、开篇 首先…

leetcode 33.搜索旋转排序数组

⭐️ 题目描述 🌟 leetcode链接:搜索旋转排序数组 ps: 本题是二分查找的变形,旋转排序数组之后其实会形成两个有序的区间。算出平均下标先判断是否与 target 相等,因为这样可以减少代码的冗余。如果前者不成立则使用平…

组合总和 II——力扣40

文章目录 题目描述法一 回溯 题目描述 法一 回溯 class Solution{ public:vector<pair<int, int>>freq;vector<vector<int>> res;vector<int> seq;void dfs(int pos, int rest){//如果目标值为0&#xff0c;说明可能有一个组合或者rest本身为0 …

(7.28-8.3)【大数据新闻速递】《数字孪生工业软件白皮书》、《中国绿色算力发展研究报告》发布;华为ChatGPT要来了

【数字孪生工业软件白皮书&#xff08;2023&#xff09;】 近日&#xff0c;第七届数字孪生与智能制造服务学术会议成功举行&#xff0c;2023《数字孪生工业软件白皮书》在会上正式发布。《白皮书》在《Digital Twin》国际期刊专家顾问委员会指导下&#xff0c;由国家重点研发计…

万字长文之 Serverless 实战详细指南

目录 前言 简易博客系统功能概要 云函数的初始化与基础配置 Tencent Serverless Toolkit for VS Code 数据库选择和设计 数据库选择 数据库设计 云函数自定义域名与 API 网关映射 域名解析 API 网关映射 云函数中的路由设计 云函数中的代码组织 Controller Servi…

搭建Django+pyhon+vue自动化测试平台

Django安装 使用管理员身份运行pycharm使用local 1 pip install django -i https://pypi.tuna.tsinghua.edu.cn/simple 检查django是否安装成功 1 python -m django --version 创建项目 1 1 django-admin startproject test cd 切换至创建的项目中启动django项目…

欧拉函数与筛法求欧拉函数

目录 欧拉函数欧拉函数的定义欧拉函数的公式欧拉函数的公式推导欧拉定理典型例题代码实现 筛法求欧拉函数思路分析经典例题代码实现 欧拉函数 欧拉函数的定义 对于任意正整数 n n n,欧拉函数 φ ( n ) φ(n) φ(n) 表示小于或等于 n n n 的正整数中&#xff0c;与 n n n …

企业网盘解析:高效的企业文件共享工具

伴随着信息技术的发展&#xff0c;越来越多的企业选择了基于云存储的企业网盘来进行企业数据存储。那么企业网盘是什么意思呢&#xff1f; 企业网盘是什么意思&#xff1f; 企业网盘&#xff0c;又称企业云盘&#xff0c;顾名思义是为企业提供的网盘服务。除了服务对象不同外&…

【maven】构建项目前clean和不clean的区别

其实很简单&#xff0c;但是百度搜了一下&#xff0c;还是没人能简单说明白。 搬用之前做C项目时总结结论&#xff1a; 所以自己在IDE里一遍遍测试程序能否跑通的时候&#xff0c;不需要clean&#xff0c;因为反正还要改嘛。 但是这个项目测试好了&#xff0c;你要打成jar包给…

全面升级:华为鸿蒙HarmonyOS4正式发布,玩趣个性化,小艺AI升级

8月4日新闻&#xff0c;今天下午&#xff0c;华为正式发布了最新版本的鸿蒙操作系统——HarmonyOS 4&#xff01; 在华为发布会上&#xff0c;鸿蒙HarmonyOS迎来了一系列令人激动的功能升级。其中包括个性化空间、多种生产力工具以及增强的手机AI助手"小艺"。这次更…

Docker入门及安装

文章目录 1.Docker概述:1.什么是docker2.为什么使用docker3.docker优点4.docker资源网址 2.Docker安装1.卸载旧版本dorcker(非必要)2.设置Docker仓库安装docker引擎4.启动验证docker卸载docker 3.Docker底层原理1.docker的结构和基本概念2.docker为什么比虚拟机快 1.Docker概述…

【新版系统架构补充】-传输介质、子网划分

传输介质 双绞线&#xff1a;无屏蔽双绞线UTP和屏蔽双绞线STP&#xff0c;传输距离在100m内 网线安装标准&#xff1a; 光纤&#xff1a;由纤芯和包层组成&#xff0c;分多模光纤MMF、单模光纤SMF 无线信道&#xff1a;分为无线电波和红外光波 通信方式和交换方式 单工…

简单易懂的生鲜蔬果小程序开发指南

随着人们对健康意识的提高&#xff0c;越来越多的人开始注重饮食健康&#xff0c;选择新鲜的果蔬产品。为了满足市场需求&#xff0c;制作一个果蔬配送小程序成为了一个不错的选择。本文将详细介绍如何快速制作一个果蔬配送小程序。 第一步&#xff1a;登录乔拓云网后台&#x…

InterProcessMutex 类的作用和使用

InterProcessMutex 类是Curator框架中的一个分布式锁的实现&#xff0c;用于在分布式环境下实现互斥锁。 InterProcessMutex 的使用步骤如下&#xff1a; 创建 CuratorFramework 客户端实例&#xff0c;并启动客户端连接到 ZooKeeper 服务器。使用 CuratorFramework 的 creat…

leetcode37. 解数独(java)

解数独 解数独题目描述回溯算法代码演示 回溯算法 解数独 难度 困难 leetcode37. 解数独 题目描述 编写一个程序&#xff0c;通过填充空格来解决数独问题。 数独的解法需 遵循如下规则&#xff1a; 1.数字 1-9 在每一行只能出现一次。 2.数字 1-9 在每一列只能出现一次。 3.数字…