.NET垃圾回收器原理及使用

.NET 应用程序中的垃圾回收器是什么?

垃圾收集器只不过是 CLR 提供的一个功能,可帮助我们清理或销毁未使用的托管对象。通过清理或销毁这些未使用的托管对象,它基本上回收内存。

当DotNet应用程序运行时,它会创建多个对象,并且在给定时刻,应用程序可能不使用其中一些对象。

因此,对于这些对象,垃圾回收器作为后台线程连续运行,并在特定的时间间隔时间,它会检查是否有任何未使用的托管对象,以及它是否发现它只是清理这些对象并回收内存。

注:垃圾回收器将仅销毁未使用的托管对象。它不清理非托管对象。

.NET垃圾回收器代数?

让我们了解什么是垃圾收集器代,它如何影响垃圾收集器的性能?

在.NET中, 有三代。它们是第0代、第1代和第2代。

了解第0代、第1代和2代

假设您有一个名为 App1 的简单应用程序。应用程序一启动,就创建 5 个托管对象。

每当创建任何新对象(新对象)时,它们都会移动到称为"第 0 代"的存储桶中。为了更好的理解,请看下图所示:

我们知道垃圾收集器作为后台线程连续运行,以检查是否有任何未使用的托管对象,以便通过清理这些对象来回收内存。

现在,假设应用程序不需要两个对象(Object1 和 Object2)。因此,垃圾回收器将销毁这两个对象(Object1 和 Object2),并回收第 0 代存储桶中的内存。

但应用程序仍然需要其余三个对象(Object3、Object4 和 Object5)。

因此,垃圾回收器不会清理这三个对象。因此,垃圾收集器将做的是,他这三个托管对象(Object3、Object4 和 Object5)将移动到第 1 代存储桶,如下图所示。

现在,假设您的应用程序又创建了两个新对象(Object6 和 Object7)。作为新对象,应在第 0 代存储桶中创建它们,如下图所示。

现在,再次运行垃圾收集器,它涉及到第 0 代存储桶和检查使用的对象。假设应用程序未使用这两个对象(Object6 和 Object7),因此它将删除这两个对象并回收内存。

现在,它转到第 1 代存储桶,并检查哪些对象未使用。假设应用程序仍然需要 Object4 和 Object5,而不需要对象 3。

因此,垃圾收集器将做什么,它将摧毁 Object3 并回收内存,以及它将 Objec4 和 Object5 移动到第 3 代存储桶,如下图所示。

什么是几代?

代不过是什么,它们将定义对象在内存中保留的时间。现在,你想到的问题是,为什么我们需要几代?

为什么我们需要几代?

通常,当我们使用大型应用程序时,它们可以创建数千个对象。因此,每个对象,如果垃圾回收器去检查他们真的是否需要,这是一个非常笨重的过程。

通过创建此类,如果第 2 代存储桶中的对象意味着"垃圾收集器"将减少对此存储桶的访问。

原因是,如果对象移动到第 2 代,则意味着它将在内存中停留更多时间。没有必要去检查他们一遍又一遍。

因此,简单地说,我们可以说第 0 代、第 1 代和 2 代有助于提高垃圾收集器的性能。第 0 代中的对象越好,性能越好,以最佳方式使用内存。

如何在类中使用析构函数,我们最终进入一个双垃圾回收器循环?

垃圾收集器将只清理托管代码。换句话说,对于任何类型的非托管代码,要清理这些代码必须由非托管代码提供,垃圾回收器无法控制它们来清理内存。

例如,假设您在 VB6 中有一个名为 MyClass 的类,然后您必须公开一些函数,例如 CLeanUp(), 在该函数中,您必须编写逻辑来清理非托管代码。

从DotNet代码中,您只需调用该方法 CLeanUp()即可启动清理。这点,或要从其中调用清理的部分是类的析构函数。

这看起来是编写清理代码的最佳地点。但是,在析构函数中编写清理时,有一个与之相关的大问题。让我们了解问题出在哪里?

在类中定义析构函数时,垃圾收集器在处置对象之前,将转到类中提出问题,您是否有析构函数,如果您有析构函数,然后将对象移动到下一代存储桶。

换句话说,即使未使用析构函数本身,它也会清理具有析构函数的对象。因此,它将等待析构函数运行,然后它会去清理对象。因此,与第 0 代相比,第 1 代和第 2 代中的对象更多。

(示例)使用析构函数

创建一个控制台应用程序,然后在程序类中复制并粘贴以下代码。

注:如果在析构函数中编写清理代码,则最终将在第 1 代和第 2 代中创建更多对象,这意味着您没有正确使用内存。

如何克服上述问题?

通过使用所谓的最终处置模式可以解决此问题。

为了实现这一点,类应实现 IDisposable 接口,并提供 Dispose 方法的实现。在 Dispose 方法中,您需要为非托管对象编写清理代码,最后需要调用 GC。通过将 true 作为输入值传递来抑制无限化(true) 方法。

此方法告诉抑制任何类型的析构函数,然后去清理对象。为了更好的理解,请看下图。

一旦您使用对象,然后您需要调用 Dispose 方法,以便双垃圾回收器循环不会发生,如下所示。

完整的代码如下:

现在,想到的问题是,为什么析构函数在那里?原因是作为开发人员,您可能忘记在使用对象后调用 Dispose 方法。在这种情况下,析构函数将调用,它将去清理对象。

到此这篇关于.NET垃圾回收器原理及使用的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • .Net 垃圾回收机制详细介绍

    析构函数 析构函数不能有修饰符,如public.不能接受任何参数. 编译器自动将一个析构函数转换成对Object.Finalize方法的一个override版,如下. class Test { protected override void Finalize() { try {-} finally { base.Finalize(); } } } 垃圾回收器 .NET垃圾回收器会保证: l  每个对象都会被摧毁,它的析构函数一定会被运行.当一个程序结束后,所有对象都会被销毁. l  每个对象只被

  • .NET垃圾回收GC诊断工具dotnet-gcmon使用

    今天介绍一个新的诊断工具 dotnet-gcmon, 也是全局 .NET CLI 工具, 它可以监控到 .NET 程序的 GC, 能获取到的信息也很详细, 另外 maoni 大佬也是其中的开发者之一. 安装 gcmon 和其他的 dotnet 诊断工具一样, 你可以使用以下命令,进行全局安装 dotnet tool install -g dotnet-gcmon 参数介绍 参数 描述 n 进程名, dotnet 应用的进程名 p 进程ID, dotnet 应用的 Process ID m 最小

  • .Net的GC垃圾回收原理及实现

    一.先了解下必备的知识前提 内存中的托管与非托管,可简单理解为: 托管:可借助GC从内存中释放的数据对象(以下要描述的内容点) 非托管:必须手工借助Dispose释放资源(实现自IDisposable)的对象 内存中有栈和堆的概念区分,仅简单说明: 栈:先进后出 的特点(这里不再详细阐述) 堆:存放数据对象实例的内存空间(以下要描述的内容点) 二..Net GC的简单描述 GC垃圾回收是对于内存堆的处理过程. 当一个应用程序进程创建时,会为此应用程序在物理内存堆中分配一块虚拟的连续性内存空间,以

  • 详谈.net中的垃圾回收机制

    1. 自动内存管理和GC 在原始程序中堆的内存分配是这样的:找到第一个有足够空间的内存地址(没被占用的),然后将该内存分配.当程序不再需要此内存中的信息时程序员需要手动将此内存释放.堆的内存是公用的,也就是说所有进程都有可能覆盖另一进程的内存内容,这就是为什么很多设计不当的程序甚至会让操作系统本身都down掉.我们有时碰到的程序莫名其妙的死掉了(随机现象),也是因为内存管理不当引起的(可能由于本身程序的内存问题或是外来程序造成的).另一个常见的实例就是大家经常看到的游戏的Trainer,他们通过

  • .NET垃圾回收器(GC)原理浅析

    作为.NET进阶内容的一部分,垃圾回收器(简称GC)是必须了解的内容.本着"通俗易懂"的原则,本文将解释CLR中垃圾回收器的工作原理. 基础知识 托管堆(Managed Heap) 先来看MSDN的解释:初始化新进程时,运行时会为进程保留一个连续的地址空间区域.这个保留的地址空间被称为托管堆. "托管堆也是堆",为什么这样说呢?这么说是希望大家不要被"术语"迷惑,这个知识点的前提是"值类型和引用类型的区别".这里假设读者已经知

  • 谈谈.net对象生命周期(垃圾回收)

    不用程序员操心的堆 -托管堆 程序在计算机上跑着,就难免会占用内存资源来存储在程序运行过程中的数据,我们按照内存资源的存取方式将内存划分为堆内存和栈内存. 栈内存,通常使用的场景是:对存取速度要求较高且数据量不大. 典型的栈内存使用的例子就是函数栈,每一个函数被调用时都会被分配一块内存,这块内存被称为栈内存,以先进后出的方式存取数据,在函数执行过程中不断往函数栈中压入(PUSH)数据(值类型数据:int.float.对象的引用...),函数执行完后又将函数栈中的数据逐个弹出(POP),由于是以操

  • .NET垃圾回收器原理及使用

    .NET 应用程序中的垃圾回收器是什么? 垃圾收集器只不过是 CLR 提供的一个功能,可帮助我们清理或销毁未使用的托管对象.通过清理或销毁这些未使用的托管对象,它基本上回收内存. 当DotNet应用程序运行时,它会创建多个对象,并且在给定时刻,应用程序可能不使用其中一些对象. 因此,对于这些对象,垃圾回收器作为后台线程连续运行,并在特定的时间间隔时间,它会检查是否有任何未使用的托管对象,以及它是否发现它只是清理这些对象并回收内存. 注:垃圾回收器将仅销毁未使用的托管对象.它不清理非托管对象. .

  • Java垃圾回收器的方法和原理总结

    什么是Java垃圾回收器 Java垃圾回收器是Java虚拟机(JVM)的三个重要模块(另外两个是解释器和多线程机制)之一,为应用程序提供内存的自动分配(Memory Allocation).自动回收(Garbage Collect)功能,这两个操作都发生在Java堆上(一段内存快).某一个时点,一个对象如果有一个以上的引用(Rreference)指向它,那么该对象就为活着的(Live),否则死亡(Dead),视为垃圾,可被垃圾回收器回收再利用.垃圾回收操作需要消耗CPU.线程.时间等资源,所以容

  • 浅谈关于Java的GC垃圾回收器的一些基本概念

    一.基本回收算法 1. 引用计数(Reference Counting) 比较古老的回收算法.原理是此对象有一个引用,即增加一个计数,删除一个引用则减少一个计数.垃圾回收时,只用收集计数为0的对象.此算法最致命的是无法处理循环引用的问题. 2. 标记-清除(Mark-Sweep) 此算法执行分两阶段.第一阶段从引用根节点开始标记所有被引用的对象,第二阶段遍历整个堆,把未标记的对象清除.此算法需要暂停整个应用,同时,会产生内存碎片. 3. 复制(Copying) 此算法把内存空间划为两个相等的区域

  • 垃圾回收器的相关知识点总结

    垃圾回收器是一把十足的双刃剑.其好处是可以大幅简化程序的内存管理代码,因为内存管理无需程序员来操作,由此也减少了(但没有根除)长时间运转的程序的内存泄漏.对于某些程序员来说,它甚至能够提升代码的性能. 另一方面,选择垃圾回收器也就意味着程序当中无法完全掌控内存,而这正是移动终端开发的症结.对于JavaScript,程序中没有任何内存管理的可能--ECMAScript标准中没有暴露任何垃圾回收器的接口.网页应用既没有办法管理内存,也没办法给垃圾回收器进行提示. 严格来讲,使用垃圾回收器的语言在性能

  • JVM垃圾回收原理解析

    概述 Java运行时区域中,程序计数器,虚拟机栈,本地方法栈三个区域随着线程的而生,随线程而死,这几个区域的内存分配和回收都具备确定性,不需要过多考虑回收问题.而Java堆和方法区则不一样,一个接口的多个实现类需要的内存不一样,一个方法的多个分支需要的内存可能也不一眼,我们只有在运行期,才能知道会创建的对象,这部分的内存分配和回收,是垃圾回收器所关注的.垃圾回收器需要完成三个问题:那些内存需要回收:什么时候回收以及如何回收. 那些垃圾需要回收 垃圾回收的基本思想是考察一个对象的可达性,即从根节点

  • JVM的7种垃圾回收器(小结)

    垃圾回收算法和垃圾回收器 对于JVM的垃圾回收算法有复制算法.标记清除.标记整理. 用阳哥的话就是:这些算法只是天上飞的理念,是一种方法论,但是真正的垃圾回收还需要有落地实现,所以垃圾回收器应运而生. JVM回收的区域包括方法区和堆,jvm对于不同区域不同的特点采用分代收集算法,比如因为所有的对象都是在Eden区进行分配,并且大部分对象的存活时间都不长,都是"朝生夕死"的,每次新生代存活的对象都不多,所以新采取复制算法:而jvm默认是新生代的对象熬过15次GC才能进入老年代,所以老年代

  • Java中垃圾回收器GC对吞吐量的影响测试

    在看内存管理术语表的时候偶然发现了"Pig in the Python(注:有点像中文里的贪心不足蛇吞象)"的定义,于是便有了这篇文章.表面上看,这个术语说的是GC不停地将大对象从一个分代提升到另一个分代的情景.这么做就好比巨蟒整个吞食掉它的猎物,以至于它在消化的时候都没办法移动了. 在接下来的这24个小时里我的头脑中充斥着这个令人窒息的巨蟒的画面,挥之不去.正如精神病医生所说的,消除恐惧最好的方法就是说出来.于是便有了这篇文章.不过接下的故事我们要讲的不是蟒蛇,而是GC的调优.我对天

  • 浅析Java中的GC垃圾回收器的意义及与GC的交互

    对象是使用new创建的,但是并没有与之相对应的delete操作来回收对象占用的内存.当我们完成对某个对象的使用时,只需停止对该对象的引用:将我们的引用改变为指向其他对象或指向null;或者从方法中返回,使得该方法的局部变量不复存在,从而使得对这些局部变量的引用变为不指向任何对象.不再被引用的对象被称为垃圾(garbage),查找并回收这些对象的过程叫做垃圾回收(garbage collection) o Java虚拟机利用垃圾回收来保证被引用的对象将会在内存中保留,同时会释放在执行代码中通过任何

随机推荐