如何利用C#实现标准的Dispose模式

作者:小菜 更新时间:2025-02-25 点击数:
简介:本文讲解的是你在建立包含内存以外资源的类型,特别是处置非内存资源的时候,如何编写自己的资源管理代码。

我们已经知道了处置那些占用非受控(unmanaged)资

【菜科解读】

本文讲解的是你在建立包含内存以外资源的类型,特别是处置非内存资源的时候,如何编写自己的资源管理代码。

我们已经知道了处置那些占用非受控(unmanaged)资源的对象的重要性,现在应该编写资源管理代码来处置那些包含非内存资源的类型了。

整个.NET框架组件都使用一个标准的模式来处理非内存资源。

使用你建立的类型的用户也希望你遵循这个标准的模式。

标准的处理模式的思想是这样的:当客户端记得的时候使用IDisposable接口释放你的非受控资源,当客户端忘记的时候防护性地使用终结器(finalizer)。

它与垃圾收集器(Garbage Collector)一起工作,确保只在必要的时候该对象才受到与终结器相关的性能影响。

这是处理非受控资源的一条很好的途径,因此我们应该彻底地认识它。

类层次体系中的根基类(root base class)必须实现IDisposable接口以释放资源。

这个类型还必须添加一个作为防御机制的终结器。

所有这些程序都把释放资源的工作委托给一个虚拟的方法,衍生的类可以根据自己的资源管理需求来重载该方法。

只要衍生的类必须释放自己的资源,并且它必须调用该函数的基类版本的时候,它才需要重载这个虚拟方法。

开始的时候,如果你的类使用了非内存资源,它就必须含有一个终结器。

你不能依赖客户端总是调用Dispose()方法。

因为当它们忘记这样做的时候,你就面临资源泄漏的问题。

没有调用Dispose是它们的问题,但是你却有过失。

用于保证非内存资源被正确地释放的唯一途径是建立终结器。

当垃圾收集器运行的时候,它立即从内存中删除所有不带终结器的垃圾对象。

所有带有终结器的对象仍然存在于内存中。

这些对象都被添加到终结队列,垃圾收集器引发一个新线程,周期性地在这些对象上运行终结器。

在这些终结程序线程完成自己的工作之后,就可以从内存中删除垃圾对象了。

需要终结的对象在内存中停留的时间比没有终结器的对象停留的时间长很多。

但是你别无选择。

如果要使程序有防护性,在类型包含非受控资源的时候,你必须编写一个终结器。

但是也不用担心性能问题。

下一步确保了客户端避免与终结相关的性能开销。

实现IDisposable接口是一种标准的途径,它通知用户和运行时系统持有资源的对象必须及时地释放。

IDisposable接口仅仅包含一个方法: public interface IDisposable { void Dispose( ); } 你对IDisposable.Dispose()方法的实现(implementation)负责下面四个事务: 1、释放所有的非受控资源。

2、释放所有的受控资源(包括未解开事件)。

3、设置标志表明该对象已经被处理过了。

你必须在自己的公共方法中检查这种状态标志并抛出ObjectDisposed异常(如果某个对象被处理过之后再次被调用的话)。

4、禁止终结操作(finalization)。

你调用GC.SuppressFinalize(this)来完成这种事务。

通过实现IDisposable接口你完成了两个事务:你为客户端及时地释放自己持有的所有受控资源提供了机制;你为客户端提供了一种释放非受控资源的标准途径。

这是一个很大的进步。

当你在类型中实现了Idisposable接口的时候,客户端可以避免终结操作的开销,你的类就成为.NET世界中的"良民"了。

但是在你建立的这种机制中仍然存在一些问题。

怎样在衍生类清理自己资源的时候同时也让基类能够清理资源?如果衍生类重载了终结操作,或者添加了自己的IDisposable实现,那么这些方法必须调用基类,否则,基类就不能正确地进行清理操作。

同样,finalize(终结操作)和Dispose参与分担了一些相同的职责。

Finalize方法和Dispose方法的代码几乎相同。

而且在重载接口函数后并不像你预料的那样工作。

标准的Dispose模式中的第三个方法是一个受保护的虚拟辅助函数,它分解出这些共同的事务,并给衍生类添加一个用于释放资源的"钩子(hook)"。

基类包含了核心接口的代码。

作为对Dispose()或终结操作的响应,该虚拟函数为衍生类清除资源提供了"钩子": protected virtual void Dispose( bool isDisposing ); 这个重载的方法实现支持finalize和Dispose的必要事务,由于它是虚拟的,它为所有的衍生类提供了一个入口点。

衍生类可以重载这个方法,为清除自己的资源提供适当的实现,同时还可以调用基类版本。

当isDisposing为真(true)的时候,你可以清除受控和非受控资源,当isDisposing为假(false)的时候,你只能清除非受控资源。

在这两种情况下,你都可以调用基类的Dispose(bool)方法,让它清除自己的资源。

下面有一个简短的例子,它演示了你在实现这种模式的时候所提供的代码框架。

MyResourceHog类演示了实现IDisposable接口、终结器的代码,并建立了一个虚拟的Dispose方法: public class MyResourceHog : IDisposable { // 已经被处理过的标记 private bool _alreadyDisposed = false; // 终结器。

调用虚拟的Dispose方法 ~MyResourceHog() { Dispose( false ); } // IDisposable的实现 // 调用虚拟的Dispose方法。

禁止Finalization(终结操作) public void Dispose() { Dispose( true ); GC.SuppressFinalize( true ); } // 虚拟的Dispose方法 protected virtual void Dispose( bool isDisposing ) { // 不要多次处理 if ( _alreadyDisposed ) return; if ( isDisposing ) { // TODO: 此处释放受控资源 } // TODO: 此处释放非受控资源。

设置被处理过标记 _alreadyDisposed = true; } } 如果衍生类需要执行另外的清除操作,它应该实现受保护的Dispose方法: public class DerivedResourceHog : MyResourceHog { // 它有自己的被处理过标记 private bool _disposed = false; protected override void Dispose( bool isDisposing ) { // 不要多次处理 if ( _disposed ) return; if ( isDisposing ) { // TODO: 此处释放受控资源 } // TODO: 此处释放所有受控资源 // 让基类释放自己的资源。

基类负责调用GC.SuppressFinalize( ) base.Dispose( isDisposing ); // 设置衍生类的被处理过标记 _disposed = true; } } 请注意,基类和衍生类都包含该对象的被处理过(disposed)标记。

这纯粹是起保护作用。

复制这个标记可以封装构成某个对象的所有类释放资源时产生的任何可能的错误。

你必须编写防护性的Dispose和finalize。

对象的处理可以按任意次序进行,你可能会遇到在调用自己类型的成员对象的Dispose()方法之前,该对象已经被处理过了。

你不应该认为这是问题,因为Dispose()方法会被多次调用。

如果它在已经被处理过的对象上被调用,它就不执行任何事务。

Finalizer(终结器)也有类似的规则。

如果你引用的对象仍然存在于内存中,你就没有必要检查空引用(null reference)。

但是,你引用的任何对象都可能被处理了,它也可能已经被终结了。

这为我带来了与处理或清除相关的任何方法的最重要的建议:你应该仅仅释放资源,在dispose方法中不要执行任何其它操作。

如果你在Dispose或finalize方法中执行其它操作,都可能给对象的生命周期带来严重的不良影响。

对象在被构造的时候才"出生",当垃圾收集器收回它们的时候才"死亡"。

当你的程序再也不能访问它们的时候,你可以认为它们处于"昏睡"状态。

如果你不能到达(reach)某个对象,你就不能调用它的方法,对于所有的意图和目的来说,它是死的。

但是带有终结器的对象被宣布死亡之前还有最后一口气。

终结器除了清理非受控资源之外不应该执行其它任何操作。

如果某个终结器由于什么原因使某个对象又可以到达了,那么该对象就恢复(resurrected)了。

即使它是从"昏睡"状态醒来的,它也是"活着"的。

下面是一个很明显的例子: public class BadClass { // 保存某个全局对象的引用 private readonly ArrayList _finalizedList; private string _msg; public BadClass( ArrayList badList, string msg ) { // 缓冲该引用 _finalizedList = badList; _msg = (string)msg.Clone(); } ~BadClass() { // 把该对象添加到列表中。

这个对象是可到达的,不再是垃圾了。

它回来了! _finalizedList.Add( this ); } } 当某个BadClass对象执行自己的终结器的时候,它向全局列表上添加了对自己的引用。

这仅仅使自己可到达了,它活了过来!但是这样操作所带来的问题使任何人都会感到胆怯。

该对象已经被终结了,因此垃圾收集器相信不用再次调用它的终结器了。

你真的需要终结一个被恢复的对象的时候,终结操作却不会发生了。

其次,你的一些资源可能不能用了。

GC不会把终结器队列中的对象可以到达的任何对象从内存中移除,但是它可能已经终结了这些对象。

如果是这样的话,那些对象一定不能再次使用了。

尽管BadClass的成员仍然存在于内存中,它们却像被处理过或被终结了一样。

在C#语言中没有控制终结次序的途径。

你不能使这种构造工作更可靠。

不要尝试! 除了学院的练习作业之外,我从来没有见到过如此明显地使用被恢复对象的代码。

但是我看到有些代码有这个倾向,它们在终结器中试图执行某些实际工作,当终结器调用的某些函数保存了对该对象的引用的时候,它就正在把对象变成活动的状态。

原则上我们必须非常仔细地检查finalizer和Dispose方法中任何代码。

如果有些代码除了释放资源之外还执行了其它的操作,我们就需要再检查一次。

这些操作在未来可能引起程序bug。

请移除这些操作,并确保finalizer和Dispose()方法只释放资源,不作其它任务事务。

在受控环境中,你不必为自己建立的每个类型编写终结器,你只需要为存储非受控类型,或者包含了实现IDisposable接口的成员的类型编写终结器。

即使你只需要Disposable接口,不需要finalizer,也应该同时实现整个模式。

否则,你会使衍生类的标准Dispose思想的实现变得很复杂,从而限制了衍生类的功能。

请遵循前面谈到的标准的Dispose思想,这将使你、你的类的用户、从你的类型建立衍生类的用户的生活更加轻松。

如何,利用,实现,标准,的,Dispose,模式,本文,

五险一金交多少钱一个月如何算?(2025

和新社-app一起来了解下:安徽池州五险一金交多少钱一个月怎么算?2024-2025年安徽池州社保费用明细参考,新社通app-社保缴费查询工具提供的最新数据如下:社保里面总共包含五种保险,不同保险缴费比例不同,具体如下:社保中的养老保险最低缴纳年限目前是180个月,即15年时间,这是享受基本养老保险待遇的“门槛”,养老保险可以累计计算缴纳年限,参保人员在不同地区参保的,其缴费年限应累计计算。

职工医保:包括统筹账户和个人账户。

单位缴纳的基本医疗保险费全部用于建立统筹基金,职工个人缴纳的基本医疗保险费,计入个人账户。

也就是说,除了统筹外,职工医保个人账户中的资金可用于在定点医疗机构门诊就医、药店购药等。

失业保险:单位缴费比例为8%-10%,个人缴费比例为2%。

同样以医疗保险缴费基数为基础,单位需缴纳的比例为8%-10%,个人则需缴纳2%。

工伤保险:由单位缴纳。

工作中(包括上下班途中)遭受意外伤害或者死亡,可以获得一大笔补偿,是对劳动者的一种保护。

生育保险费由企业按月缴纳,职工个人不缴纳生育保险费。

具体比例根据不同类型的单位有所不同:事业单位按在职职工缴纳基本养老保险费的基数作为缴纳生育保险费的基数,按照0.4%的比例缴纳生育保险费。

一般企业按在职职工缴纳基本养老保险费的基数作为缴纳生育保险费的基数,按照0.8%的比例缴纳生育保险费。

社保缴费更改缴费档次如何改?交社保有啥好处?(2025/04/03)

社保交的档次可以更改吗?怎么调整社保缴费档次?社保缴费更改缴费档次怎么改?交社保有啥好处?如下:社保档次是可以更改的。

社保缴费档次的更改主要涉及到缴费基数的调整,这通常会影响到个人的社保待遇和未来的社保福利。

社保档次更改根据相关法律法规,如《中华人民共和国社会保险法》,个人和用人单位都有权依法缴纳社会保险费,并有权查询缴费记录、个人权益记录等。

同时,社会保险基金包括基本养老保险基金、基本医疗保险基金、工伤保险基金、失业保险基金和生育保险基金,且这些基金应专款。

社保档次更改的流程如果是有工作单位的个人,想要更改社保缴费档次,可以直接委托公司的社保专员携带缴费工资基数变更材料,到当地的社保部门办理更改手续。

如果是个人自行缴纳的社保,个人可以带上社保卡、身份证等资料亲自去社保部门办理更改手续。

/社保档次更改的时机与限制通常,社保缴费档次的更改可以在每年的特定时间段内进行,如每年的5月份,但具体时间可能因地区而异。

此外,社保缴费档次每年只能更改一次,因此需要抓紧时间办理。

总的来说,按月交的社保档次是可以根据个人的实际情况进行更改的,但更改过程需要遵循一定的要求和流程,且每年只有一次更改机会。

好处要想知道社保有没有必要交,我们先来了解一下社保的作用都有哪些。

社保包含养老保险、医疗保险、生育保险、工伤保险、失业保险五个险种。

当我们缴纳了社保后,可以享受以下保-障:1.经济保险:社保中的各项保险体系,如养老保险、医疗保险、失业保险等,能够为参保人在遭遇风险时提供经济上的支持。

2.社会保险:社保体系还体现了对公民的社会责任,通过为公民提供基本的生活和福利,促进社会公平和稳定。

缴纳社保是公民履行社会责任的体现,也是享受社会保险体系的前提。

3.储蓄和投资:社保体系中的公积金体系,为公民提供了一种储蓄和投资的方式。

通过缴纳公积金,公民可以在购房、租房等方面获得资金支持,同时也可以通过公积金的增值收益获得一定的投资回报。

温馨提示:本数据源于网络,仅供参考!具体需以当地具体法规为准!

加入收藏
               

如何利用C#实现标准的Dispose模式

点击下载文档

格式为doc格式

  • 账号登录
社交账号登录