.Net Framework中,把资源分为托管资源和非托管资源两大类,数据库
托管资源指能够经过.Net Frame垃圾回收器进行回收的资源,主要是指分配在托管堆上你的内存资源,这类资源的回收是不须要人工干预,.Net Framework的垃圾回收器会在合适的时刻进行回收,程序也能够主动调用GC.Collect()强制执行垃圾回收(可是不建议)。编程
非托管资源则是指不能被.Net Framework垃圾回收器回收的资源,主要包括如下几类:文件读写、窗口操做、网络链接、数据库链接、GDI画刷、图标等。针对这类资源,垃圾回收器可以跟踪其生存期,但不了解具体如何清理这些资源。针对这类资源,垃圾回收器在清理的时候回调用Object.Finalize()方法,该方法是虚方法,非托管对象须要重写方法来实现资源回收。网络
托管对象是不能重载Object.Finalize()方法,编译器会自动根据析构函数生成对应的Object.Finalize()方法,所以,当托管对象中使用到非托管资源时,须要在析构函数中释放该资料。app
析构函数的定义以下(在方法名称为类名称前加了一个“~”):函数
public class TestA { public string Name = "TestA"; ~TestA() { Console.WriteLine("Dispose TestA"); } }
因为析构函数是有垃圾回收器主动调用的,所以,调用该方法前,类包含的托管对象可能已经被主动回收了,此时再进行释放操做,可能会形成异常,所以,不建议在析构函数中释放托管资源。性能
垃圾回收器的操做时机是.NetFramework本身所决定的,它会在系统内存的占用和系统系统之间做一个平衡来决定什么时候出发,所以,对于非托管资源,这类资源属于宝贵资源,必须及时地释放掉,不能有垃圾回收器来决定什么时候释放。spa
.Net Framework中为回收资源,而定义了一个接口,IDisposable定义为:code
1 // 2 // 摘要: 3 // Defines a method to release allocated resources. 4 [ComVisible(true)] 5 public interface IDisposable 6 { 7 // 8 // 摘要: 9 // Performs application-defined tasks associated with freeing, releasing, or resetting 10 // unmanaged resources. 11 void Dispose(); 12 }
任何包括非托管资源的类都须要继承此接口,并在接口方法中实现对托管资源和非托管资源的释放操做,同时在析构函数中实现对非托管资源的释放。orm
这么实现的好处就是:代码中显示调用Dispose()时,可以主动地释放资源(托管和非托管),垃圾回收器就会移除该对象不会执行回收操做(即再也不执行Finalize()方法),从而提升了性能。而若是没有显示调用Dispose(),垃圾回收器也会在适当的时机调用Finalize()方法来释放非托管资源。对象
C#语言中using关键字做为语法糖出现,将非托管资源的使用包含在using()中使用,执行完成后会主动调用Dispose方法来释放资源。
基于性能考虑,应该下降使用析构函数来释放资源的使用频率,所以,没有析构函数的对象在垃圾处理器中直接删除,而包括析构函数的对象,则须要先执行析构函数,而后删除对象,增长了一次操做,会下降垃圾回收器的工做效率,从而影响性能。所以,对于非托管对象的释放,应该实现IDispose接口来回收资源,而不依赖垃圾回收器。
当前类使用new关键字建立了另外一个类的实例时,当前类对新建的类实例的引用称为强引用,当新建类实例没有释放时,垃圾回收器则没法回收这个对象。只有将该引用释放的时候,垃圾回收器才会回收该对象。
1 TestA a = new TestA(); 2 a = null; 3 GC.Collect(); 4 //当把变量a置为null,主动执行垃圾回收后就会回收该对象。
编程过程,咱们会遇到这样的场景,持有一个对象实例的引用,这个实例比较庞大且建立成本不是过高,可是使用频率比较低。该状况下,能够使用WeakReference对这个实例进行包装,当须要使用当对象时,先调用WeakReference.IsAlive是否等于true或者Target不等于null,判断该对象实例是否存在,若存在则调用其Target并强制转换后使用,若不存在,则须要从新建立一个新的实例来使用。使用WeakReference包括的对象实例,垃圾回收器会释放该实例,回收其内存空间。所用,使用前必须判断其是否仍然存在。