本文实例讲述了C#中IDisposable模式的用法,针对垃圾资源的回收进行了较为详细的讲解。分享给大家供大家参考之用。具体要领如下:
首先,对付垃圾回收而言,在C#中,托管资源的垃圾回收是通过CLR的Garbage Collection来实现的,Garbage Collection会挪用仓库上东西的析构函数完成东西的释放事情;而对付一些非托管资源,好比数据库链接东西等,需要实现IDisposable接口进行手动的垃圾回收。那么什么时候使用Idisposable接口,以及如何使用呢?
先来参考一下如下代码:
public interface IDisposable { void Dispose(); } public class DisposablClass : IDisposable { //是否回收完毕 bool _disposed; public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } ~DisposableClass() { Dispose(false); } //这里的参数暗示示是否需要释放那些实现IDisposable接口的托管东西 protected virtual void Dispose(bool disposing) { if(_disposed) return; //如果已经被回收,就中断执行 if(disposing) { //TODO:释放那些实现IDisposable接口的托管东西 } //TODO:释放非托管资源,设置东西为null _disposed = true; } }
Dispose()要领
当需要回收非托管资源的DisposableClass类,就挪用Dispoase()要领。而这个要领不会被CLR自动挪用,需要手动挪用。
~DisposableClass(),,析构函数
当托管堆上的东西没有被其它东西引用,GC会在回收东西之前,挪用东西的析构函数。这里的~DisposableClass()析构函数的意义在于报告GC你可以回收我,Dispose(false)暗示在GC回收的时候,就不需要手动回收了。
虚要领Dispose(bool disposing)
通过此要领,所有的托管和非托管资源都能被回收。参数disposing暗示是否需要释放那些实现IDisposable接口的托管东西。
如果disposings设置为true,就暗示DisposablClass类依赖某些实现了IDisposable接口的托管东西,可以通过这里的Dispose(bool disposing)要领挪用这些托管东西的Dispose()要领进行回收。
如果disposings设置为false,就暗示DisposableClass类依赖某些没有实现IDisposable的非托管资源,那就把这些非托管资源东西设置为null,期待GC挪用DisposableClass类的析构函数,把这些非托管资源进行回收。
此外,以上
把Dispose(bool disposing)要领设置为protected virtual的原因是但愿有子类可以一起参预到垃圾回收逻辑的设计,而且还不会影响到基类
。好比有这样的一个子类:
public class SubDisposableClass : DiposableClass { private bool _disposed; //暗示是否已经被回收 protected override void Dispose(bool disposing) { if(!_disposed) //如果还没有被回收 { if(disposiing) //如果需要回收一些托管资源 { //TODO:回收托管资源,挪用IDisposable的Dispose()要领就可以 } //TODO:回收非托管资源,把之设置为null,期待CLR挪用析构函数的时候回收 _disposed = true; } base.Dispose(disposing);//再挪用父类的垃圾回收逻辑 } }
在.NET 2.0之前,如果一个东西的析构函数抛出异常,这个异常会被CLR忽略。但.NET 2.0以后,如果析构函数抛出异常就会导致应用措施的瓦解。所以,保证析构函数不抛异常变得非常重要
。
还有,Dispose()要领允许抛出异常吗?答案是否定的。
如果Dispose()要领有抛出异常的可能,那就需要使用try/catch来手动捕获
。以下是考虑Dispose()要领有异常可能的写法:
public class DisposableClass : IDisposable { bool _disposed; ...... protected virtual void Dispose(bool disposing) { if(_disposed) return; if(disposing) { //TODO:挪用托管资源的Dispose()要领进行垃圾回收 } try { _channelFactory.Close(); //*的时候可能会有异常 } catch(Exception ex) { _log.Warn(ex);//记录日志 try { _channelFactory.Abort();//抛弃的时候可能会有异常 } catch(Exception cex) { _log.Warn(cex);//记录日志 } } _channelFactory = null; _disposed = true; } }
总结:
当我们自界说的类及其业务逻辑中引用某些托管和非托管资源,就需要实现IDisposable接口,实现对这些资源东西的垃圾回收。