首页 > 编程 > C# > 正文

C#基础:Dispose()、Close()、Finalize()的区别详解

2020-01-24 03:16:15
字体:
来源:转载
供稿:网友

.net内存回收与DisposeCloseFinalize方法
一. net的对象使用一般分为三种情况
1.创建对象
2.使用对象
3.释放对象
二.创建对象
1.创建对象实际分为两个步骤变量类型宣告和初始化对象
2.变量类型宣告(declare),如

复制代码 代码如下:

FileStream fs

这行代码会在当前的变量作用域空间(栈或堆)里建立一个叫做fs的变量至少四个字节吧(因为要存一个对象的地址)
3.初始化对象
对象在使用(调用其方法或属性)前必须进行初始化。

复制代码 代码如下:

fs = new FileStream(@"C:/test.txt",FileMode.OpenOrCreate);

这行代码会分成3个步骤
a.在托管堆中分配一块内存其大小等于FileStream中所有字段(当然不包括静态的)的内存总和加上MS认为需要的其它东东。
b.初始化对象的字段(值类型的把其位全部初始化成0,对象初始化为null当然string是一个例外它被初始化成空字符串)
c.调用FileStream相应的构造器这里会初始化一个非托管资源(文件)的私有字段。
三.使用对象
使用对象就没什么讲的就是调用对象的方法(或属性等)来完成某个功能当然为了释放对象而调用的方法其范畴应不属于此类中(现在提到的Finalize等)
四.释放对象
1.释放对象也就是说这个对象我已经不需要了现在我要把其释放以便把其在堆上所占用的内存空间给收回来(当然变量名的内存空间就不需要管了因为它会随其作用域自动消失)
2. .net自动进行内存管理也就是说当它判断一个对象没有用了(当然有自己的算法)它就会将其内存给自动收回来但是其收回的时间一般不确定(当.net认为内存紧张时它就会开始)
BTW:其实我们就是想自己收回对象的内存也不可能因为MS没有提供途径(GC.Collect也是启动.net的内存收集功能)
五.第一个结论
在net中使用对象很简单创建对象之后直接使用就可以了不用了也不要去管它垃圾收集器会帮你把内存要回来的。
六.例外
当对象的成员引用了一个非托管资源时(不在托管堆上分配的内存或资源像文件数据库连接等等)下面以一个例子来说明
System.IO.FileStream类别这是.net基本类库提供的一个非托管资源(文件)封装对象(用Reflector工具反编译mscorlib.dll可见其代码)
1.FileStream毫无疑问封装了一个非托管资源
观其源代码发现有这样一个私有成员
复制代码 代码如下:

private SafeFileHandle _handle;

通过构造器调用的Init方法可以发现这个成员的初始化代码
复制代码 代码如下:

this._handle = Win32Native.SafeCreateFile(text2, num1, share, secAttrs, mode, num2, 
Win32Native.NULL);

而后者实际上就是kernel32.dll中的CreateFile方法它返回一个HANDLE(即非托管资源引用)
2.我们先来使用这个类别
复制代码 代码如下:

using System;
using System.IO;
public class TestFileStream
{
    public static void Main(string[] args)
    {   
           //创建一个FileStream对象
        FileStream fs = new FileStream(@"C:/test.txt",FileMode.OpenOrCreate);       
        Console.WriteLine("您可以尝试在系统中删除c盘下的test.txt(回车键继续)");
        //暂停程序执行并尝试在系统中删除那个文件
        Console.ReadLine();

        //删除文件测试
        try
        {
            File.Delete(@"c:/test.txt");
        }
        catch (IOException ex)
        {
            Console.WriteLine("[Error]程序删除文件失败{0}",ex.Message);
        }
    }
}


3.在程序挂起时(Console.ReadLine等待输入)删除文件会失败很容易理解因为文件打开后没有将其关闭系统不知道这个文件是否还有用所以帮我们保护这个文件(理所当然那个非托管资源所使用的内存还被程序占用着)
4.但是在程序执行完后我们再尝试删除文件成功为什么?(fs不是没有关闭那个SafeFileHandle吗?)
当然您可以说windows操作系统在一个进程结束后会自动回收其资源没错(但是如果是com就惨了因为com是存在于自己的独立进程内而操作系统不负责这个:(   )不过这里不是因为windows操作系统的功能而是.net垃圾收集器帮的忙。
5.看下面这个例子
复制代码 代码如下:

using System;
using System.IO;
public class TestFileStream
{
    public static void Main(string[] args)
    {
        //创建一个FileStream对象
        FileStream fs = new FileStream(@"C:/test.txt", FileMode.OpenOrCreate);
        Console.WriteLine("您可以尝试在系统中删除c盘下的test.txt(回车键继续)");
       //暂停程序执行并尝试在系统中删除那个文件
        Console.ReadLine();

        /**//*进行垃圾收集*/
        GC.Collect();
        Console.WriteLine("再删一下试试");
        Console.ReadLine();
    }
}


6.注意中间那行代码:
复制代码 代码如下:

GC.Collect();

这是强制要.net垃圾收集器进行垃圾收集。
我们再去尝试删除test.txt居然可以被删除了为什么呀?(fs不是没有关闭那个SafeFileHandle吗?)让我细细道来
7.我们首先了解一下.net垃圾收集器进行垃圾收集的四种时机(参见.net框架程序设计 李建忠译)
a.最常见的当.net觉得合适时例如它感到内存紧张了(语称为0代对象充满)
b.微软强烈不建议使用的GC的Collect方法调用(就是我们上面用的这种啦因为会降低性能会挂起进程, 等等反正听微软的吧。当然某些时候可以用就像我上面用来测试的代码呵呵...)
c.应用程序域卸载时(AppDomain)
d.CLR被关闭时
8.现在我们可以明白第1个例子为什么在程序结束后文件可以被删除因为CLR被关闭时.net执行了垃圾收集(也就是等同于第二个例子的GC.Collect()代码)
9.所以现在所有的问题都集中到垃圾收集上面它做了什么?
a.垃圾收集器在判断一个对象不会再被引用到后就开始对它进行垃圾收集(即回收内存)
b.清空内存(即把托管堆中的内存收回来)
c.但是对象的有些字段引用到了非托管资源怎么办?如FileStream的_handle
d.所以我们必须告诉垃圾收集器在你回收我的内存之前先帮我执行一个方法来收回我的非托管资源以免托管堆的内存被你回收了而我引用的非托管资源的内存却被泄漏了。
e.这个方法就是Finalize()也就是C#的 ~ClassName() 方法(同C++中的析构语法)
f.所以一个对象如果存在Finalize方法时垃圾收集器在收回它的内存之前就会自动调用这个方法
g.这样我们就可以把那些东东(非托管资源)给清理干净了
由此看来垃圾收集器提供的这种机制就是为了更好的完善.net的自动内存管理的功能让我们也可以参与到垃圾收集中去
10.我们再来看看GC.Collect()这行代码或CLR关闭时.Net做了什么
a.垃圾收集器启动发现fs引用的那个对象已经没用了(当然CLR关闭时才不管你有没有用通通回收)于是对它进行内存回收
b.发现fs的类型FileStream提供了Finalize方法于是先调用这个方法
(以下通过Reflector继续)
c.Finalize方法中有 this._handle.Dispose()代码于是调用SafeHandler.Dispose()
d.接着转到(当然好多个圈您悠着点...)SafeFileHandle.ReleaseHandle方法发现代码Win32Native.CloseHandle() (即关闭非托管资源--文件HANDLE)
真相大白原来是垃圾收集器帮我们关闭了那个非托管资源(当然还是通过我们自己写的Finalize方法)因此后面就可以删除文件了。
11.有人会问好像我们平时在使用FileStream对象时没这么复杂呀?
答Very Good!
一部分人是因为大家都和我的例1一样有好运气那个C盘下的test.txt文件自从被创建后我压根就不会再去用它管它这部分资源有没有被泄漏有没有被锁定最后程序结束时被垃圾收集器帮了忙把忘了关闭的文件HANDLE给收回来了。
剩下的一部分人在程序里埋下了一颗"哑弹"不知什么时候会爆炸就像我例子中的File.Delete方法就出现了异常。
(不过我觉得)绝大多数人是在看了很多诸如.net编程忠告Microsoft强烈建议MSDN标准做法等等等等( 还有我这篇blog呵呵)之后知道了在使用如FileStream,SqlConnection这些东东时必须将其Close。
12.Close与Dispose
查看我们那两个例子的代码都是不标准的正确做法应该在使用完那个FileStream后调用fs.Close()将其关闭以保证资源的安全。
附正确做法
复制代码 代码如下:

using System;
using System.IO;
public class TestFileStream
{
    public static void Main(string[] args)
    {
        //创建一个FileStream对象
        FileStream fs = new FileStream(@"C:/test.txt", FileMode.OpenOrCreate);
        /**//*在用完FileStream后关闭*/
        fs.Close();

        //删除文件测试
        try
        {
            File.Delete(@"c:/test.txt");
        }
        catch (IOException ex)
        {
            Console.WriteLine("[Error]程序删除文件失败{0}", ex.Message);
        }
    }
}


13.有人举手讲这么多早告诉我调用fs.Close不就得了。
哥们fs.Close()方法是由您写的调不调用手在您身上您不调用的话哪天程序出了问题您有会叫微软真垃圾.net真不稳定还是java好安全可靠...    为防您的国骂MS只好在垃圾收集中加这一款以防不测...
14.Dispose模式
认真查看.net类库中的那些基本类别凡是有Finalize方法的类别基本上都提供了诸如Dispose,Close,Dispose(bool)等方法(FileStream也不例外)
15.其实不管是Dispose,Close,Finalize方法最终应该都是执行相同的代码
区别
Finalize方法只能由微软调用
Dispose和Close方法提供给您调用
因此在您使用完那些类别后那就直接调用Close吧(没有Close再调用Dispose方法)当然万一您忘了也别担心还有垃圾收集器帮您垫后。
七.第二个结论
1.在您开发一个封装非托管资源(即类中的字段引用到了非托管资源)的类别时
A:强烈建议您提供Finalize方法进行非托管资源的释放.net垃圾收集器不会帮您自动回收那部分资源而是通过调用您的Finalize方法来帮您释放。(这样可以保证在使用您类别的那位程序员忘了手动回收内存时还可通过垃圾收集器来补救)
B.强烈建议您提供一个Close或Dispose方法以便使用您类别的程序员可以手动释放您的类别中的非托管资源。(参见.net框架程序设计 自动内存管理一章实现Dispose模式)
C.如果类别封装了像FileStream这样的对象(即对非托管资源的再次封装)时一般也应该提供一 个Close或Dispose方法除非您的这个成员保证在每次使用后都被正常的关闭即对调用者透明。
2.在您使用一个封装非托管资源的类别时
A:强烈建议您在明确知道这个类别没有用之后调用其提供的Close或Dispose方法手动释放其非托管资源的 内存。有道是有借有还再借不难;借了不还再借休想~~
B:注意在手动释放后不要再调用该对象的相关方法了因为对象已经损毁了
再次BTW:不管是FinalizeClose还是Dispose您都无法显式释放托管堆内存它们永远是微软的"私人财产 ")
总结:Dispose 和 Close基本上是一样的。 Close 是为那些不熟悉Dispose的开发者设计的,Close让人更容易理解是做什么的。
在.net framework 里面,close()被设计成public的,并且在close()里面调用被隐藏的dispose(),而后dispose()再去调用另一个virtual的dispose(bool)
函数。所以如果从这个class继承,你就必须实现dispose(bool)方法 。调用者通过close()就会间接调用到你重载的那个dispose(bool)方法去释放资源。
因为close()只是用来呼叫那个隐藏的dispose()继而呼叫dispose(bool)的,用户不应该改变close行为的
具有disponse()方法的类是实现 了IDisposable 接口的。 .net中有很多class 只提供close(),而不对外提供disponse(),但是它的确实现了idisponse接口,
这是为什么呢?
究其原因是因为接口的实现模式 ---- 显式实现 隐式实现,二者的区别:对于 隐式实现 来说,你只需要调用 “new ClassA().Dispose()“,但是对于显式实现
来说,dispose()不会是这个 classA 的成员函数。唯一的调用方式是先强制类型转换到 IDisposable ,即"new ClassA().Dispose()",但是((IDisposable)new ClassA()).Dispose() 可以编译过。这样就符合了设计的要求:提供 close(),隐藏dispose(),并且实现 IDisposeable接口

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表