如何预防Net下的资源泄露呢?有一种方法就是继承IDisposable接口,再对继承中的所有实例递归地调用Finalize方法。
(1)一个原则:资源在哪个类中被创建就应该在哪个类中清理。如果类中调用了其他基类中创建的资源,则应在基类中删除这些资源。
(2)析构函数:
析构函数是由垃圾回收器(GC)在清理对象时调用的。
因为.NET中的托管对象都是由垃圾回收器自动定期清理的,所以如果一个类中只有托管对象,则垃圾回收器在回收该对象时会同时一次性清理掉该类中创建的托管对象,此种情况下不要编写析构函数(情况A)。
如果一个类中创建使用了非托管资源(如数据库连接)(情况B),此时应该使用析构函数,但也只是作为忘记调用Dispose()函数的一种备份机制。换言之,此时,应该先掉用Dispose()函数来删除资源。
(3)Dispose()函数:
在上面的情况A中,可以不调用Dispose()函数。但如果类中创建使用过一些较大的托管对象,最好尽快清除它们,此时可以在Dispose函数中删除它们,并由用户调用以尽快删除它们。
在情况B中,应该在Dispose()中删除非托管资源,并由用户调用Dispose()。此时,为防止垃圾回收器再次调用析构函数,应该在Dispose()中调用GC.SuppressFinalize(this)通知垃圾回收器,此对象已经不再需要执行析构函数以免重复执行。但如果用户忘记了调用Dispose(),则垃圾回收器仍然会执行析构函数,保证非托管资源会被清除。
具体实现方法参见下文例子:
public class MyClass():IDisposable
{
private StreamReader sr;
private int connection;
......
public void Dispose()
{
Dispose(true);
GC.SuppressFinally(this);
}
protected virtual void Dispose(bool disposing)
{
if(disposing)
{ //清理托管对象
if(sr!=null)
{
sr.Close();
sr=null;
}
}
//清理非托管对象
CloseConnection();//假设类中有这样一个函数用于清理DB连接这个非托管资源
}
~MyClass()
{
Dispose(false); //仅仅清理非托管资源,除此外不应编写其它代码。
}
.....
}
另外有一点,不应该重写Finalize()函数。按照面向对象的原理来看,如果你重写了一个基类函数但没有使用override,它将隐藏基类中的函数,可能产生非预期的结果与冲突。
Net如何继承IDisposable接口,实现自己的Dispose()函数
最新推荐文章于 2023-04-14 11:13:11 发布