托管资源和非托管资源
托管资源:.net可以自动进行回收的资源,主要是指托管堆上分配的内存资源。托管资源的回收工作不需要人工干预的,有.net运行库在合适时调用垃圾回收器进行回收。
非托管资源:是.net不知道如何回收的资源,最常见的一类非托管资源是包装操作系统资源的对象,如文件,窗口,网络连接,数据库连接,画刷,图标等。这类资源,垃圾回收器在清理的时候会调用Object.Finalize()方法。默认情况下,方法是空的,对于非托管对象,需要在此方法中编写回收非托管资源的代码,以便垃圾回收器正确回收资源。在.net中,Object.Finalize()方法是无法重载的,编译器是根据类的析构函数来自动生成Object.Finalize()方法的,所以对于包含非托管资源的类,可以将释放非托管资源的代码放在析构函数。
注意,不能在析构函数中释放托管资源,因为析构函数是有垃圾回收器调用的,可能在析构函数调用之前,类包含的托管资源已经被回收了,从而导致无法预知的结果。
使用析构方法来对非托管资源进行回收,不能保证及时的释放非托管资源,因为垃圾回收器是由CRL自动调用的。因此定义了一个Dispose()方法,让使用者可以手动的释放非托管资源和托管资源。Dispose()方法是由使用者调用的,在调用时,类的托管资源和非托管资源肯定都未被回收,所以可以同时回收两种资源。
Microsoft为非托管资源的回收专们定义了一个接口:IDisposable,接口中包含一个Dispose()方法,任何包含非托管资源的类,都应该继承此接口。
在一个包含非托管资源的类中,关于资源释放的标准做法是:
1.继承IDisposable接口;
2.实现Dispose()方法,在其中释放托管资源和非托管资源,并将对象本身从垃圾回收器中移除(垃圾回收器不再回收此资源)
3.实现类析构函数,在其中释放非托管资源。
在使用时,显示调用Dispose()方法,可以及时的释放资源,同时通过移除Finalize()方法的执行,提高了性能。
在.net中应该尽可能的少用析构函数释放资源。没有析构函数的对象在垃圾处理器一次处理中从内存删除,但有析构函数的对象,需要两次,第一次调用析构函数,第二次删除对象。而且在析构函数中包含大量的释放资源代码,会降低垃圾回收器的工作效率,影响性能,所以对于包含非托管资源的对象,最好及时的调用Dispose()方法来回收资源,而不是依赖垃圾回收器。
1 using System; 2 using System.Collections.Generic; 3 using System.ComponentModel; 4 using System.Linq; 5 using System.Text; 6 using System.Threading.Tasks; 7 8 namespace Apps.Models 9 { 10 public class BaseResource:IDisposable 11 { 12 13 //结构,属于非托管资源 14 private IntPtr handle; 15 //组件,属于托管资源 16 private Component comp; 17 #region IDisposable Support 18 private bool disposedValue = false; // 要检测冗余调用【是否已释放资源的标志】 19 20 public BaseResource() 21 { 22 23 } 24 25 protected virtual void Dispose(bool disposing) 26 { 27 if (!disposedValue) 28 { 29 if (disposing) 30 { 31 // TODO: 释放托管状态(托管对象)。 32 comp.Dispose(); 33 } 34 35 // TODO: 释放未托管的资源(未托管的对象)并在以下内容中替代终结器。 36 // TODO: 将大型字段设置为 null。 37 handle = IntPtr.Zero; 38 39 disposedValue = true; 40 } 41 } 42 // TODO: 仅当以上 Dispose(bool disposing) 拥有用于释放未托管资源的代码时才替代终结器。 43 ~BaseResource() 44 { 45 // 请勿更改此代码。将清理代码放入以上 Dispose(bool disposing) 中。 46 Dispose(false); 47 } 48 49 // 添加此代码以正确实现可处置模式。 50 public void Dispose() 51 { 52 // 请勿更改此代码。将清理代码放入以上 Dispose(bool disposing) 中。 53 Dispose(true); 54 // TODO: 如果在以上内容中替代了终结器,则取消注释以下行。 55 GC.SuppressFinalize(this); 56 } 57 #endregion 58 59 } 60 }
参考资料: