欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

C#基础知识回顾:1.由WeakReference想到对象的创建与销毁

程序员文章站 2022-06-23 22:03:04
.Net Framework中,把资源分为托管资源和非托管资源两大类, 托管资源指可以通过.Net Frame垃圾回收器进行回收的资源,主要是指分配在托管堆上你的内存资源,这类资源的回收是不需要人工干预,.Net Framework的垃圾回收器会在合适的时刻进行回收,程序也可以主动调用GC.Coll ......

.Net Framework中,把资源分为托管资源和非托管资源两大类,

托管资源指可以通过.Net Frame垃圾回收器进行回收的资源,主要是指分配在托管堆上你的内存资源,这类资源的回收是不需要人工干预,.Net Framework的垃圾回收器会在合适的时刻进行回收,程序也可以主动调用GC.Collect()强制执行垃圾回收但是不建议

非托管资源则是指不能被.Net Framework垃圾回收器回收的资源,主要包括以下几类:文件读写、窗口操作、网络连接、数据库连接、GDI画刷、图标等。针对这类资源,垃圾回收器能够跟踪其生存期,但不了解具体如何清理这些资源。针对这类资源,垃圾回收器在清理的时候回调用Object.Finalize()方法,该方法是虚方法,非托管对象需要重写方法来实现资源回收。

托管对象是不能重载Object.Finalize()方法,编译器会自动根据析构函数生成对应的Object.Finalize()方法,因此,当托管对象中使用到非托管资源时,需要在析构函数中释放该资料。

析构函数的定义如下(在方法名称为类名称前加了一个“~”):

    public class TestA
    {
        public string Name = "TestA";
        ~TestA()
        {
            Console.WriteLine("Dispose TestA");
        }
    }

 

由于析构函数是有垃圾回收器主动调用的,因此,调用该方法前,类包含的托管对象可能已经被主动回收了,此时再进行释放操作,可能会造成异常,因此,不建议在析构函数中释放托管资源。

垃圾回收器的操作时机是.NetFramework自己所决定的,它会在系统内存的占用和系统系统之间作一个平衡来决定何时出发,因此,对于非托管资源,这类资源属于宝贵资源,必须及时地释放掉,不能有垃圾回收器来决定何时释放。

.Net Framework中为回收资源,而定义了一个接口,IDisposable定义为:

 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     }

任何包括非托管资源的类都需要继承此接口,并在接口方法中实现对托管资源和非托管资源的释放操作,同时在析构函数中实现对非托管资源的释放。

这么实现的好处就是:代码中显示调用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包括的对象实例,垃圾回收器会释放该实例,回收其内存空间。所用,使用前必须判断其是否仍然存在。