前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >C# 托管资源与非托管资源

C# 托管资源与非托管资源

作者头像
zls365
发布于 2020-12-29 04:19:40
发布于 2020-12-29 04:19:40
3.5K00
代码可运行
举报
文章被收录于专栏:CSharp编程大全CSharp编程大全
运行总次数:0
代码可运行

托管资源:

Net平台中,CLR为程序员提供了一种很好的内存管理机制,使得程序员在编写代码时不要显式的去释放自己使用的内存资源(这些在先前C和C++中是需要程序员自己去显式的释放的)。这种管理机制称为GC(garbage collection)。GC的作用是很明显的,当系统内存资源匮乏时,它就会被激发,然后自动的去释放那些没有被使用的托管资源(也就是程序员没有显式释放的对象)。

所以托管就是.net framework 负责帮你管理内存及资源释放,不需要自己控制,当然对象只针对托管资源(部分引用类型), 不回收非托管资源。 像数组,用户定义的类、接口、委托,object,字符串等引用类型,栈上保存着一个地址而已,当栈释放后, 即使对象已经没有用了,但堆上分配的内存还在,只能等GC收集时才能真正释放 ;但注意int,float,DateTime之类的值类型,GC会自动释放他们占用的内存,不需要GC来回收释放

那么非托管的资源怎么释放回收呢?

非托管资源:

对于非托管资源,GC只能跟踪非托管资源的生存期,而不知道如何去释放它。这样就会出现当资源用尽时就不能提供资源能够提供的服务,windows的运行速度就会变慢。比如当你链接了数据库,用完后你没有显式的释放数据库资源,如果还是不断的申请数据库资源,那么到一定时候程序就会抛出一个异常。

所以,当我们在类中封装了对非托管资源的操作时,我们就需要显式,或者是隐式的释放这些资源在.Net中释放非托管资源主要有2种方式,Dispose,Finalize,而Finalize和Dispose方法分别就是隐式和显式操作中分别使用到的方法。

例如文件流,数据库的连接,系统的窗口句柄,打印机资源等等,当你读取文件之后,就需要对各种Stream进行Dispose等操作。比如 SqlDataReader 读取数据完毕之后,需要 reader.Dispose();等

Finalize一般情况下用于基类不带close方法或者不带Dispose显式方法的类,也就是说,在Finalize过程中我们需要隐式的去实现非托管资源的释放,然后系统会在Finalize过程完成后,自己的去释放托管资源。在.NET中应该尽可能的少用析构函数释放资源,MSDN2上有这样一段话:实现 Finalize 方法或析构函数对性能可能会有负面影响,因此应避免不必要地使用它们。用 Finalize 方法回收对象使用的内存需要至少两次垃圾回收。所以有析构函数的对象,需要两次,第一次调用析构函数,第二次删除对象。而且在析构函数中包含大量的释放资源代码,会降低垃圾回收器的工作效率,影响性能。所以对于包含非托管资源的对象,最好及时的调用Dispose()方法来回收资源,而不是依赖垃圾回收器。

托管资源指的是.NET可以自动进行回收的资源,主要是指托管堆上分配的内存资源。托管资源的回收工作是不需要人工干预的,有.NET运行库在合适调用垃圾回收器进行回收。

非托管资源指的是.NET不知道如何回收的资源,最常见的一类非托管资源是包装操作系统资源的对象,例如文件,窗口,网络连接,数据库连接,画刷,图标等。这类资源,垃圾回收器在清理的时候会调用Object.Finalize()方法。默认情况下,方法是空的,对于非托管对象,需要在此方法中编写回收非托管资源的代码,以便垃圾回收器正确回收资源。

在.NET中,Object.Finalize()方法是无法重载的,编译器是根据类的析构函数来自动生成Object.Finalize()方法的,所以对于包含非托管资源的类,可以将释放非托管资源的代码放在析构函数。

注意,不能在析构函数中释放托管资源,因为析构函数是有垃圾回收器调用的,可能在析构函数调用之前,类包含的托管资源已经被回收了,从而导致无法预知的结果。

本来如果按照上面做法,非托管资源也能够由垃圾回收器进行回收,但是非托管资源一般是有限的,比较宝贵的,而垃圾回收器是由CRL自动调用的,这样就无法保证及时的释放掉非托管资源,因此定义了一个Dispose()方法,让使用者能够手动的释放非托管资源。Dispose()方法释放类的托管资源和非托管资源,使用者手动调用此方法后,垃圾回收器不会对此类实例再次进行回收。Dispose()方法是由使用者调用的,在调用时,类的托管资源和非托管资源肯定都未被回收,所以可以同时回收两种资源。

Microsoft为非托管资源的回收专门定义了一个接口:IDisposable,接口中只包含一个Dispose()方法。任何包含非托管资源的类,都应该继承此接口。

在一个包含非托管资源的类中,关于资源释放的标准做法是:

(1) 继承IDisposable接口;

(2) 实现Dispose()方法,在其中释放托管资源和非托管资源,并将对象本身从垃圾回收器中移除(垃圾回收器不在回收此资源);

(3) 实现类析构函数,在其中释放非托管资源。

在使用时,显示调用Dispose()方法,可以及时的释放资源,同时通过移除Finalize()方法的执行,提高了性能;如果没有显示调用Dispose()方法,垃圾回收器也可以通过析构函数来释放非托管资源,垃圾回收器本身就具有回收托管资源的功能,从而保证资源的正常释放,只不过由垃圾回收器回收会导致非托管资源的未及时释放的浪费。

在.NET中应该尽可能的少用析构函数释放资源。在没有析构函数的对象在垃圾处理器一次处理中从内存删除,但有析构函数的对象,需要两次,第一次调用析构函数,第二次删除对象。而且在析构函数中包含大量的释放资源代码,会降低垃圾回收器的工作效率,影响性能。所以对于包含非托管资源的对象,最好及时的调用Dispose()方法来回收资源,而不是依赖垃圾回收器。

上面就是.NET中对包含非托管资源的类的资源释放机制,只要按照上面要求的步骤编写代码,类就属于资源安全的类。

下面用一个例子来总结一下.NET非托管资源回收机制:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
      Public class BaseResource:IDisposable
         {
                  PrivateIntPtr handle; // 句柄,属于非托管资源
                  PrivateComponet comp; // 组件,托管资源
                  Privateboo isDisposed = false; // 是否已释放资源的标志
        
                  PublicBaseResource
                  {
                  }
        
                  //实现接口方法
                  //由类的使用者,在外部显示调用,释放类资源
                  Publicvoid Dispose()
                  {
                            Dispose(true);// 释放托管和非托管资源
                           
                            //将对象从垃圾回收器链表中移除,
                            // 从而在垃圾回收器工作时,只释放托管资源,而不执行此对象的析构函数
                            GC.SuppressFinalize(this);
                  }
        
                  //由垃圾回收器调用,释放非托管资源
                  ~BaseResource()
                  {
                            Dispose(false);// 释放非托管资源
                  }
        
                  //参数为true表示释放所有资源,只能由使用者调用
                  //参数为false表示释放非托管资源,只能由垃圾回收器自动调用
                  //如果子类有自己的非托管资源,可以重载这个函数,添加自己的非托管资源的释放
                  //但是要记住,重载此函数必须保证调用基类的版本,以保证基类的资源正常释放
                  Protectedvirtual void Dispose(bool disposing)
                  {
                            If(!this.disposed)// 如果资源未释放 这个判断主要用了防止对象被多次释放
                            {
                                     If(disposing)
                                     {
                                              Comp.Dispose();// 释放托管资源
                                     }
                           
                                     closeHandle(handle);// 释放非托管资源
                                     handle= IntPtr.Zero;
                            }
                            this.disposed= true; // 标识此对象已释放
                  }
         }

析构函数只能由垃圾回收器调用。

Despose()方法只能由类的使用者调用。

在C#中,凡是继承了IDisposable接口的类,都可以使用using语句,从而在超出作用域后,让系统自动调用Dispose()方法。 一个资源安全的类,都实现了IDisposable接口和析构函数。提供手动释放资源和系统自动释放资源的双保险。

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-12-27,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 CSharp编程大全 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
C#内存泄漏的成因、检测与预防策略
C#作为一种托管语言,虽由.NET框架的垃圾回收机制自动管理内存,但仍存在多种内存泄漏的可能性。内存泄漏会导致应用程序内存占用持续增长,最终引发性能下降甚至崩溃。在C#程序中,主要的内存泄漏原因包括事件订阅未取消、静态变量持有引用、非托管资源未释放以及匿名函数闭包等。针对这些潜在问题,开发者可以通过使用Visual Studio内存分析器或JetBrains dotMemory等工具进行检测,并采取相应的预防措施。本文将系统分析内存泄漏的成因、检测方法及预防策略,帮助开发者提升C#程序的内存管理能力。
代码小李
2025/05/16
1100
从C#垃圾回收(GC)机制中挖掘性能优化方案
GC,Garbage Collect,中文意思就是垃圾回收,指的是系统中的内存的分配和回收管理。其对系统性能的影响是不可小觑的。今天就来说一下关于GC优化的东西,这里并不着重说概念和理论,主要说一些实用的东西。关于概念和理论这里只做简单说明,具体的大家可以看微软官方文档。
跟着阿笨一起玩NET
2018/09/20
2K0
从C#垃圾回收(GC)机制中挖掘性能优化方案
.NET面试题系列[5] - 垃圾回收:概念与策略
面试出现频率:经常出现,但通常不会问的十分深入。通常来说,看完我这篇文章就足够应付面试了。面试时主要考察垃圾回收的基本概念,标记-压缩算法,以及对于微软的垃圾回收模板的理解。知道什么时候需要继承IDisposible接口,解构函数是做什么用的,什么时候需要自己写一个解构函数。
s055523
2018/09/14
9610
改善C#程序的建议4:C#中标准Dispose模式的实现
需要明确一下C#程序(或者说.NET)中的资源。简单的说来,C#中的每一个类型都代表一种资源,而资源又分为两类:
潘成涛
2019/05/25
7220
【C# 基础精讲】构造函数和析构函数
构造函数(Constructor)和析构函数(Destructor)是面向对象编程中的两个重要概念,它们分别用于在对象创建和销毁的时候执行特定的操作。这两个函数在C#以及其他面向对象编程语言中都具有重要的作用,用于初始化对象的状态、分配资源,以及在对象被销毁时释放资源。在本文中,我们将详细探讨构造函数和析构函数的概念、特点、用法以及在C#中的实际应用。
繁依Fanyi
2023/10/12
5050
【C# 基础精讲】构造函数和析构函数
ASP.NET Core 中的内存管理和垃圾回收 (GC)
GC 会分配堆段,其中每个段都是一系列连续的内存。 置于堆中的对象归类为 3 个代系之一:0、1 或 2。 代系可确定 GC 尝试在应用不再引用的托管对象上释放内存的频率。 编号较低的代系会更加频繁地进行 GC。 对象会基于其生存期从一个代系移到另一个代系。 随着对象生存期延长,它们会移到较高代系。 如前所述,较高代系进行 GC 的频率较低。 短期生存的对象始终保留在第 0 代中。 例如,在 Web 请求存在期间引用的对象的生存期较短。 应用程序级别单一实例通常会迁移到第 2 代。 当 ASP.NET Core 应用启动时,GC 会:
用户9857551
2023/10/17
4370
ASP.NET Core 中的内存管理和垃圾回收 (GC)
C#-垃圾回收机制(GC)
The garbage collector is a common language runtime component that controls the allocation and release of managed memory。
郑子铭
2023/08/29
2.3K0
C#-垃圾回收机制(GC)
C# IDispose
在C#中,IDisposable 是一个接口,用来提供一种机制来释放未使用的资源。当对象持有非托管资源(例如文件句柄、数据库连接、网络套接字等)时,需要实现 IDisposable 接口。
JusterZhu
2023/10/24
2470
C# IDispose
C#:单例,闭包,委托与事件,线程,Parallel,Params,扩展方法,接口与抽象类
在对泛型的约束中,最常使用的关键字有where 和 new。 其中where关键字是约束所使用的泛型,该泛型必须是where后面的类,或者继承自该类。 new()说明所使用的泛型,必须具有无参构造函数,这是为了能够正确的初始化对象
立羽
2024/01/15
4090
[C#] 利用using与try/finally来清理资源
如果某个类型用到了非托管型的系统资源,那么就需要通过IDisposable接口的Dispose()方法来明确地释放。.NET环境规定,这种资源并不需要由包含该资源的类型或系统来释放,而是应该由使用此类型的代码释放。也就是说,如果你使用了带有Dispose()方法的类型,那么就应该调用它的Dispose()方法以释放其中的资源,而要想确保该方法总是能够得到调用,最好的办法就是利用using语句或try/finally代码块。
科控物联
2022/03/29
8570
[C#] 利用using与try/finally来清理资源
【深入浅出C#】章节10: 最佳实践和性能优化:内存管理和资源释放
C#对象池示例代码: 以下是一个简单的C#对象池示例,用于管理字符串对象。注意,这只是一个示例,实际应用中可以根据需要自定义更复杂的对象池。
喵叔
2023/09/11
1.5K0
熟悉而陌生的新朋友——IAsyncDisposable
在.NET Core 3.0的版本更新中,官方我们带来了一个新的接口 IAsyncDisposable。
句幽
2021/09/08
8090
谈谈.net对象生命周期
程序在计算机上跑着,就难免会占用内存资源来存储在程序运行过程中的数据,我们按照内存资源的存取方式将内存划分为堆内存和栈内存。
梁规晓
2019/12/17
1.4K0
谈谈.net对象生命周期
黑马公开课——运行原理与GC学习笔记
.NET Framework的组成: (1)基础类库(BCL):使用线程的类来完成编程,对于不存在的类,就自己编写; (2)编译工具:将源文件,编译成“程序集”(exe或dll等)[.NET环境中,MSIL=CIL=IL] (3)公共语言运行时(CLR):执行前检测、编译;执行到了某个方法时才编译这个方法的代码[即时编译器(JIT)] 编译过程:.NET源代码(C#)——>通过C#编译器编译成程序集[程序集中包括:元数据(一个表,显示了程序中有什么成员,类,字段,方法等),IL代码等资源] 运行例子: (1)源代码如下: using System;
Edison Zhou
2018/08/20
4610
黑马公开课——运行原理与GC学习笔记
【深入浅出C#】章节 4: 面向对象编程基础:构造函数和析构函数
构造函数和析构函数是面向对象编程中的两个重要概念,它们在对象的创建和销毁过程中起着关键作用。 构造函数是一个特殊的成员函数,用于在创建对象时初始化对象的数据成员。它的主要作用是为对象分配内存空间并初始化对象的状态。构造函数具有与类同名的特点,并且没有返回类型。通过构造函数,可以确保对象在创建时具有有效的初始状态。构造函数可以被重载,这意味着可以根据需要定义多个具有不同参数的构造函数。 析构函数是一个特殊的成员函数,用于在对象销毁时执行必要的清理操作。它的主要作用是释放对象占用的资源,例如释放动态分配的内存、关闭打开的文件或释放其他外部资源。析构函数的名称与类名相同,前面加上一个波浪线(~)作为前缀。析构函数在对象销毁时自动调用,无法手动调用。 构造函数和析构函数在对象的生命周期中起着关键作用。构造函数确保对象在创建时具有合适的初始化状态,而析构函数则确保对象在销毁时进行必要的清理操作。这种对象创建和销毁的过程对于程序的正确运行和资源管理非常重要。合理使用构造函数和析构函数可以提高代码的可读性、可维护性和可靠性,同时避免内存泄漏和资源泄漏等问题。
喵叔
2023/07/09
8820
.Net中Finalize()和Dispose()有什么区别?
Finalize自动释放资源,Dispose()用于手动释放资源。 释放类所使用的未托管资源的两种方式: 1.利用运行库强制执行的析构函数,但析构函数的执行是不确定的,而且,由于垃圾收集器的工作方式,它会给运行库增加不可接受的系统开销。 2.IDisposable接口提供了一种机制,允许类的用户控制释放资源的时间,但需要确保执行Dispose()。 一般情况下,最好的方法是执行这两种机制,获得这两种机制的优点,克服其缺点。假定大多数程序员都能正确调用Dispos
程序你好
2018/07/20
1.6K0
析构函数(C#)
 析构函数又称终结器,用于析构类的实例。 定义   析构函数(destructor) 与构造函数相反,当对象结束其生命周期时(例如对象所在的函数已调用完毕),系统自动执行析构函数。析构函数往往用来做“清理善后” 的工作(例如在建立对象时用new开辟了一片内存空间,delete会自动调用析构函数后释放内存)。 析构函数简介 以C++语言为例:[1]  析构函数名也应与类名相同,只是在函数名前面加一个位取反符~,例如~stud( ),以区别于构造函数。它不能带任何参数,也没有返回值(包括void类型)。只能有一
房上的猫
2018/03/14
1.9K0
C# 客户端内存优化分析
C# 开发客户端系统的时候,.net 框架本身就比较消耗内存资源,特别是xp 这种老爷机内存配置不是很高的电脑上运行,所以就需要进行内存上的优化,才能流畅的在哪些低端电脑上运行. 想要对C# 开发的客户端内存优化需要了解以下几个概念。
Jlion
2022/04/07
1.2K0
C# 中的内存管理与垃圾回收机制
内存管理是计算机编程中的核心问题之一。在C#中,内存的分配与释放由系统自动管理,减轻了开发者手动管理内存的负担。这主要归功于C#的垃圾回收(Garbage Collection,GC)机制。本文将详细介绍C#的内存管理模式与垃圾回收机制,帮助开发者更深入地理解其原理和优化应用性能的方法。
Michel_Rolle
2024/09/14
2.8K0
如何实现标准的dispose
前面的文章我们说过,如果对象包含非托管资源那么就必须要正确的清理,现在我们就来说一下如何清理。针对非托管资源 .NET 会采用一套标准的模式来完成清理工作。也就是说如果开发人员自己编写的类中存在非托管资源,那么这个类的使用者就会认为这个类遵循 .NET 的垃圾清理模式。标准的 dispose 模式即实现了 IDisposable 接口,又实现了 finalizer ,这样就可以在客户端忘记调用 IDisposable.Dispose 的情况下也可以释放资源。
喵叔
2020/09/08
8490
相关推荐
C#内存泄漏的成因、检测与预防策略
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验