服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C#中Dispose和Finalize方法使用介绍

C#中Dispose和Finalize方法使用介绍

2023-02-23 15:08.NET开发菜鸟 C#

这篇文章介绍了C#中Dispose和Finalize方法的用法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

一、前言

在C#中,由于有了垃圾回收机制的支持,对象的析构和以前的C++有了很大的不同,这就要求程序员在设计类型的时候,充分理解.NET的机制,明确怎样利用Dispose方法和Finalize方法来保证一个对象正确而高效地被析构。

二、Dispose方法的功能

我们在讲解有关using的用法时,已经介绍了Dispose方法。正是因为垃圾回收机制掩盖了对象内存真正被回收的时间,考虑到很多情况下程序员扔希望在对象不再被使用的时候进行一些清理工作,所以.NET提供了IDisposable接口并且在其中定义了Dispose方法。通常程序员会在Dispose方法中实现一些托管对象和非托管对象的释放以及逻辑业务的结束工作等。注意实现了Dispose方法不能得到任何有关释放的保证,Dispose方法的调用依赖于类型的使用者,当类型被不恰当地使用时,Dispose方法将不会被调用,但using等语法的存在还是帮助了类型的Dispose方法被调用。

三、Finalize方法的机制

由于Dispose方法的调用依赖于使用者,为了弥补这一缺陷,.NET同时提供了Finalize方法。Finalize方法常常被具有C++开发经验的程序员称为析构方法,但它的执行方法却和传统C++中的析构函数完全不同。Finalize方法在GC执行垃圾回收时调用,具体的机制是这样的:

  • 当每个包含Finalize方法的类型的实例对象被分配时,.NET会在一张特定的表结构中添加一个引用并且执行这个实例对象。方便起见称该表为“带析构对象表”。
  • 当GC执行并且检测到一个不被使用的对象时,需要进一步检查“带析构对象表”来查看该对象类型是否有Finalize方法,如果没有则该对象被视为垃圾,如果存在Finalize方法,则把指向该对象的引用从“带析构对象表”移到另外一张表中,这里暂时称它为“等待析构表”。并且该对象实例被视为扔在被使用。
  • CLR将有一个单独的线程负责处理“等待析构表”,其方法就是依次通过引用调用其中每个对象的Finalize方法,然后删除引用,这时托管堆中的对象实例将处于不再被使用的状态。
  • 下一个GC执行时,将释放已经被调用Finalize方法的那些对象实例。

四、正确地使用Dispose和Finalize方法

Finalize方法确实比Dispose方法更加安全,因为它由CLR保证调用,但是性能方面Finalize方法却要差的多。我们需要知道的是:正确的类型设计是把Finalize方法作为Dispose方法的后备,只有在使用者没有调用Dispose方法的情况下,Finalize方法才能被视为需要执行。下面是一个正确高效的设计模板,建议牢记这个模板并且套用到每一个需要DIspose和Finalize方法的类型上去。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
using System;
 
namespace usingDemo
{
    public class FinalizeDisposeBase : IDisposable
    {
        // 标记对象是否已被释放
        private bool _disposed = false;
        // Finalize方法
        ~FinalizeDisposeBase()
        {
            Dispose(false);
        }
 
        /// <summary>
        /// 这里实现了IDisposable中的Dispose方法
        /// </summary>
        public void Dispose()
        {
            Dispose(true);
            // 告诉GC此对象的Finalize方法不再需要调用
            GC.SuppressFinalize(true);
        }
 
        /// <summary>
        /// 在这里做实际的析构工作
        /// 声明为虚方法以供子类在必要时重写
        /// </summary>
        /// <param name="isDisposing"></param>
        protected virtual void Dispose(bool isDisposing)
        {
            // 当对象已经被析构时,不在执行
            if(_disposed)
            {
                return;
            }
            if(isDisposing)
            {
                // 在这里释放托管资源
                // 只在用户调用Dispose方法时执行
            }
            // 在这里释放非托管资源
            // 标记对象已被释放
            _disposed = true;
        }
    }
 
    public sealed class FinalizeDispose:FinalizeDisposeBase
    {
        private bool _mydisposed = false;
        protected override void Dispose(bool isDisposing)
        {
            // 保证只释放一次
            if (_mydisposed)
            {
                return;
            }
            if(isDisposing)
            {
                // 在这里释放托管的并且在这个类型中声明的资源
            }
            // 在这里释放非托管的并且在这个类型中声明的资源
            // 调用父类的Dispose方法来释放父类中的资源
            base.Dispose(isDisposing);
            // 设置子类的标记
            _mydisposed = true;
        }
        static void Main()
        {
 
        }
    }
}

上面的代码是一个近乎完美的Dispose配合Finalize的设计模板,其中有几点需要特别注意:

  • 真正做释放工作的只是Virtual的受保护方法Dispose方法,事实上这个方法的名字并不重要,仅仅为了通用和更好理解,称呼它为Dispose。
  • 虚方法Dispose需要接受一个布尔类型的参数,主要用于区分调用方是类型的使用者还是.NET的垃圾回收。前者通过IDisposable的Dispose方法,而后者通过Finalize方法。两者的区别是通过Finalize方法释放资源时不能再释放或使用对象中的托管资源,这是因为这时的对象已经处于不被使用的状态,很有可能其中的托管资源已经被释放掉了。
  • 在IDisposable的Dispose方法的实现中通过GC.SuppressFinalize()方法来告诉.NET此对象在被回收时不需要调用Finalize方法,这一句是改善性能的关键,记住实现Dispose方法的本质目的就是避免所有释放工作在Finalize方法中进行。
  • 子类型必须定义自己的释放标记来标明子类中的资源是否已经被释放,同时子类的虚方法Dispose方法也只需要释放自己新定义的资源。
  • 确保在虚方法Dispose中做的都是释放工作,有些逻辑上的结束工作需要反复斟酌,以防止一个简单的赋值语句使对象再度存活。

五、总结

Dispose方法被使用者主动调用,而Finalize方法在对象被垃圾回收的第一轮回收后,由一个专用的.NET线程进行调用。Dispose方法不能保证被执行,而.NET的垃圾回收机制保证了拥有Finalize方法并且需要被调用的类型对象的Finalize方法被执行。调用Finalize方法涉及了一系列复杂的操作,性能代价非常高,程序员可以通过GC.SuppressFinalize方法通知.NET该对象的Finalize方法不需要被调用。有关Dispose和Finalize的类型设计应该参照上面的代码模板,以保证对象能够被高效和安全的释放。

到此这篇关于C#中Dispose和Finalize方法的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://www.cnblogs.com/dotnet261010/p/12330981.html

延伸 · 阅读

精彩推荐
  • C#C#读写文本文件(.txt)的方法实例

    C#读写文本文件(.txt)的方法实例

    读写文本文件其实是件很简单的事情,这篇文章主要给大家介绍了关于C#读写文本文件(.txt)的相关资料,需要的朋友可以参考下...

    君莫笑·秋7672022-11-17
  • C#C#实现的自定义邮件发送类完整实例(支持多人多附件)

    C#实现的自定义邮件发送类完整实例(支持多人多附件)

    这篇文章主要介绍了C#实现的自定义邮件发送类,具有支持多人多附件的功能,涉及C#邮件操作的相关技巧,需要的朋友可以参考下...

    Autumoon5232021-11-05
  • C#区分c# 前台和后台线程

    区分c# 前台和后台线程

    这篇文章主要介绍了c# 前台线程和后台线程的区别与联系,文中讲解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下...

    团队buff工具人7472022-09-23
  • C#Unity实现物体弧线运动到规定的坐标

    Unity实现物体弧线运动到规定的坐标

    这篇文章主要为大家详细介绍了Unity实现物体以弧线的形式运动到规定的坐标,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可...

    天人合一moonlight11782022-09-16
  • C#C#语言MVC框架Aspose.Cells控件导出Excel表数据

    C#语言MVC框架Aspose.Cells控件导出Excel表数据

    这篇文章主要为大家详细介绍了C#语言MVC框架Aspose.Cells控件导出Excel表数据,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    筱筱脱脱9392022-03-06
  • C#NPOI实现两级分组合并功能(示例讲解)

    NPOI实现两级分组合并功能(示例讲解)

    下面小编就为大家分享一篇NPOI实现两级分组合并功能的示例讲解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    Tiger.liang10922022-02-15
  • C#C#很简单而又很经典的一句代码实例

    C#很简单而又很经典的一句代码实例

    这篇文章主要给大家分享介绍了关于C#很简单而又很经典的一句代码,文中通过示例代码介绍的非常详细,对大家学习或者使用C#具有一定的参考学习价值,...

    不三周助9992022-07-29
  • C#深入了解C#多线程安全

    深入了解C#多线程安全

    使用多线程无法避免的一个问题就是多线程安全。那什么是多线程安全?如何解决多线程安全?本文将通过一些简单的例子为大家详细介绍一下多线程相关...

    小六公子6422022-12-13