首页 > 学院 > 开发设计 > 正文

C#中标准Dispose模式的实现

2019-11-17 02:34:27
字体:
来源:转载
供稿:网友
C#中标准Dispose模式的实现

本文引自:[Luminji]http://www.VEVb.com/luminji/archive/2011/03/29/1997812.html

需要明确一下C#程序(或者说.NET)中的资源。简单的说来,C#中的每一个类型都代表一种资源,而资源又分为两类:

托管资源:由CLR管理分配和释放的资源,即由CLR里new出来的对象;

非托管资源:不受CLR管理的对象,windows内核对象,如文件、数据库连接、套接字、COM对象等;

毫无例外地,如果我们的类型使用到了非托管资源,或者需要显式释放的托管资源,那么,就需要让类型继承接口IDisposable。这相当于是告诉调用者,该类型是需要显式释放资源的,你需要调用我的Dispose方法。

不过,这一切并不这么简单,一个标准的继承了IDisposable接口的类型应该像下面这样去实现。这种实现我们称之为Dispose模式:

复制代码publicclass SampleClass : IDisposable { //演示创建一个非托管资源PRivate IntPtr nativeResource = Marshal.AllocHGlobal(100); //演示创建一个托管资源private AnotherResource managedResource =new AnotherResource(); privatebool disposed =false; ///<summary>/// 实现IDisposable中的Dispose方法 ///</summary>publicvoid Dispose() { //必须为true Dispose(true); //通知垃圾回收机制不再调用终结器(析构器) GC.SuppressFinalize(this); } ///<summary>/// 不是必要的,提供一个Close方法仅仅是为了更符合其他语言(如C++)的规范 ///</summary>publicvoid Close() { Dispose(); } ///<summary>/// 必须,以备程序员忘记了显式调用Dispose方法 ///</summary>~SampleClass() { //必须为false Dispose(false); } ///<summary>/// 非密封类修饰用protected virtual /// 密封类修饰用private ///</summary>///<param name="disposing"></param>protectedvirtualvoid Dispose(bool disposing) { if (disposed) { return; } if (disposing) { // 清理托管资源if (managedResource !=null) { managedResource.Dispose(); managedResource =null; } } // 清理非托管资源if (nativeResource != IntPtr.Zero) { Marshal.FreeHGlobal(nativeResource); nativeResource = IntPtr.Zero; } //让类型知道自己已经被释放 disposed =true; } publicvoid SamplePublicMethod() { if (disposed) { thrownew ObjectDisposedException("SampleClass", "SampleClass is disposed"); } //省略 } }复制代码

在Dispose模式中,几乎每一行都有特殊的含义。

在标准的Dispose模式中,我们注意到一个以~开头的方法:

复制代码///<summary>/// 必须,以备程序员忘记了显式调用Dispose方法 ///</summary>~SampleClass() { //必须为false Dispose(false); }复制代码

这个方法叫做类型的终结器。提供终结器的全部意义在于:我们不能奢望类型的调用者肯定会主动调用Dispose方法,基于终结器会被垃圾回收器调用这个特点,终结器被用做资源释放的补救措施。

一个类型的Dispose方法应该允许被多次调用而不抛异常。鉴于这个原因,类型内部维护了一个私有的布尔型变量disposed:

private bool disposed = false;

在实际处理代码清理的方法中,加入了如下的判断语句:

if (disposed) { return; } //省略清理部分的代码,并在方法的最后为disposed赋值为true disposed =true;

这意味着类型如果被清理过一次,则清理工作将不再进行。

应该注意到:在标准的Dispose模式中,真正实现IDisposable接口的Dispose方法,并没有实际的清理工作,它实际调用的是下面这个带布尔参数的受保护的虚方法:

复制代码///<summary>/// 非密封类修饰用protected virtual /// 密封类修饰用private ///</summary>///<param name="disposing"></param> protectedvirtualvoid Dispose(bool disposing) { //省略代码 }复制代码

之所以提供这样一个受保护的虚方法,是为了考虑到这个类型会被其他类继承的情况。如果类型存在一个子类,子类也许会实现自己的Dispose模式。受保护的虚方法用来提醒子类必须在实现自己的清理方法的时候注意到父类的清理工作,即子类需要在自己的释放方法中调用base.Dispose方法。

还有,我们应该已经注意到了真正撰写资源释放代码的那个虚方法是带有一个布尔参数的。之所以提供这个参数,是因为我们在资源释放时要区别对待托管资源和非托管资源。

在供调用者调用的显式释放资源的无参Dispose方法中,调用参数是true:

publicvoid Dispose() { //必须为true Dispose(true); //其他省略 }

Word-break: normal; word-spacing: 0px; margin: 10px auto; letter-spacing: normal; line-height: 16px; background-color: #ffffff; text-indent: 0px; -webkit-text-stroke-w

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表