析构方法:

我们知道引用类型都有构造方法(constructor),相对应的也有一个析构方法(destructor).顾名思义,构造方法,就是在创建这个对象时,要执行的方法。例如,我们可以通过构造方法,

初始化字段。析构方法,就是当这个对象被垃圾回收后(garbage collected,我们称回收对象内存为垃圾回收 garbage collection),要执行的方法。关于析构方法,需要大家注意的是,垃圾回收一个对象,并不是析构方法完成的(下面会讲到垃圾回收的工作原理),析构方法只有在对象被垃圾回收后才执行。也就是说,析构方法对于一个对象来讲,不是必须的。很多时候,如果加上它,反而不好(下面讲garbage collector怎样工作时,就会明白不好的原因)。

既然垃圾回收不归析构方法负责,那么它有什么用呢?因为垃圾回收是CLR自动执行的,CLR只能处理受管理资源(managed resource),那些不受管理资源(unmanaged resource)就需要我们

自己去处理了。例如文件读取(file stream),当对象结束时,我们需要把文件流关掉。关掉文件流的代码,就要在析构方法中。也就是说,析构方法的用处,在处理不受管理资源时用处比较大。
看个例子:

class FileProcessor
{
FileStream file=null;
public FileProcess(string fileName)
{
this.file=file.OpenRead(fileName); //open file for reading
} ~FileProcess() //析构方法与构造方法很相似,不同的是析构方法要加一个~
{
this.file.Close(); //close file
}
}

file对象,是CRL垃圾回收,当file被垃圾回收后,运行析构方法FileProcess,此时我们将非管理资源关掉。this.file.Close().

这里有几条对析构方法的限制:

1.只有引用类型才可以有析构方法。

struct MyStruct(){
~MyStruct(){....}//结构是值类型,所以不能有析构方法
}

2.不能对析构方法提供访问修饰符.

public ~FileProcessor(){};//错误的

3.析构方法不可以有参数.

~FileProcessor(int param)
{
....// //错误的
};

之所以会有这三条限制,是因为析构方法只能由CLR调用,自己不可以调用。因为,你不知道引用对象什么时候被垃圾回收了,只有对象被垃圾回收了,程序才会自己调用析构方法。

在内部,程序会将我们写的析构方法,转换一下。例如:

class FileProcessor
{
~fileProcessor(){...} //析构方法
} class FileProcessor
{ protected override void Finalize()
{
try{ }
finally{
base.Finalize(); //CLR会将析构方法转变成这个
}
}
}

我们把执行析构方法的过程,称为结束(finalization,或终结。)

垃圾回收机制(garbage collector)

我们上边提到,回收内存空间,回收不用的引用类型对象的过程称为垃圾回收(garbage collection).这个过程是由CLR通过Garbage collector这样一个机制去运行的。

当我们在程序中创建变量,会在内存中开辟一段空间。电脑的内存不是无限大的,我们需要在变量超越定义的范围(程序不再需要这个变量了)时,对它所占的内存进行管理,处理这些内存。当变量不再被使用时,需要把内存回收。值类型变量回收内存,非常简单。

当它超出定义的范围时就会自动被毁掉,被占的内存也会自动回收。超出定义的范围,指的是当它不再被使用,不能再被使用。引用类型变量回收内存,比较麻烦。例如:

fileProcessor myFp=new fileProcessor();
fileProcessor referenceToMyFp=myFp;

想一下这种情况,myFp对象已经超出定义的范围。此时我们去回收内存,要把myFp引用堆上的内存回收。可是,恰恰此时,referenceToMyFp还在引用准备回收的内存,如果此时把内存回收,当程序运行referenceToMyFp时,程序就会出错。所以,只用当所有引用对象都超出定义的范围时,也就是都不再使用时,才可以去回收这些对象引用的内存。确保程序中这些指向同一块儿的引用对象全部不再使用,是很困难,很复杂的.所以C#设计者,把处理引用类型回收内存的工作,交给了CLR(Common language running).CLR利用garbage collector机制,来处理这些事情。

垃圾回收机制工作原理

garbage collector在自己的线程中工作,在特定的时间执行。一般,当程序运行到一个方法的最后时,就会工作。它工作时,其他线程就会暂时停止工作。因为,garbage collector可能会移除或者更新对象引用。

1.garbage collector会创建一张表,表里存放所有的可得到对象(reachable objects,.可得到对象,说白了就是指那些还在使用,不能回收内存的对象。)。

2.检查一下那些不可得到对象(unreachable objects,就是那些超出定义范围,需要回收内存的对象),看看他们是否有析构方法。(destructor),如果有,就把这些对象放入一个叫做

freachable queue的队列里。

3.把那些不可得到对象,且没有析构方法的对象所指向的内存地址回收。它是通过将那些可得对象在堆上的地址下移,这样堆上面就留出了可用的内存。此时,garbage collector也会更新堆

上的引用地址。(因为,地址有变化)。

4.此时,程序中其他的线程恢复工作。

5.garbage collector 通过调用自己的Finalize方法来结束不可得到对象,且有析构方法的对象。(前面我们讲了,析构方法不是必须的,有时候会给程序带来复杂,累赘。如果,没有析构方法,当

CLR运行Garbage collector时,第五步就可以省去)。

资源管理

有些资源很稀缺,稀缺到不能等到CLR去调用析构方法去处理。例如database connections,file handles.此时,我们就需要写一个dispose方法,手动去处理资源。(dispose可以换成任何名

字,这里只是举个例子)。例如:

TextReader reader=new StreamReader(filename);
string line;
while((line=reader.ReadLine())!=null)
{
Console.WriteLine(line);
}
reader.Close();

这里,Close就是一个dispose方法,手动去关掉文件流。但是,这样有个问题,当出现异常时,有可能导致reader.Close()不被执行,这样资源一直被占用。所以,我们要改写成:

TextReader reader=new StreamReader(filename);
try
{
string line;
while((line=reader.ReaderLine())!=null)
{
Console.WriteLine(line);
}
}
finally
{
reader.Close();
}

这样无论如何,reader.Close()都会被执行,资源都会被释放。不过即便改写成这样,也不是最完美的。因为,当我们执行完finally块儿内的代码,也就是在try finally之后,我们有可能

无意中使用reader对象,就是释放掉资源后的reader对象。此时,我们可以用using 语句来解决这些不足。我们将上面的代码改成using之后:

using(TextReader reader=new StreamReader(filename))
{
string line;
while((line=reader.ReadLine())!=null)
{
Console.Writle(line);
}
}

执行完using之后,资源自动释放,越过using范围,对象不能用。一个对象如果要被支持使用using,该对象必须实现IDispose接口。

我们自己创建一个类,继承IDispose接口,让它可以用在USING语句中。这里我们应该区分一下析构方法,与dispose方法。我们知道析构方法一定会执行,但是不知道什么时候执行。我们

知道dispose方法什么时候执行,但是不知道它会不会执行,因为类里有dispose方法,不代表一定会把这个类用在using语句中。此时,我们就可以通过析构方法来调用dispose方法,这样可

以保证dispose方法一定被调用。

C#垃圾回收的更多相关文章

  1. Javascript 的执行环境(execution context)和作用域(scope)及垃圾回收

    执行环境有全局执行环境和函数执行环境之分,每次进入一个新执行环境,都会创建一个搜索变量和函数的作用域链.函数的局部环境不仅有权访问函数作用于中的变量,而且可以访问其外部环境,直到全局环境.全局执行环境 ...

  2. 修改session垃圾回收几率

    <?php //修改session垃圾回收几率 ini_set('session.gc_probability','1'); ini_set('session.gc_divisor','2'); ...

  3. .NET面试题系列[5] - 垃圾回收:概念与策略

    面试出现频率:经常出现,但通常不会问的十分深入.通常来说,看完我这篇文章就足够应付面试了.面试时主要考察垃圾回收的基本概念,标记-压缩算法,以及对于微软的垃圾回收模板的理解.知道什么时候需要继承IDi ...

  4. .net垃圾回收机制编程调试试验

    1. 什么是CLR GC? 它是一个基于引用跟踪和代的垃圾回收器. 从本质上,它为系统中所有活跃对象都实现了一种引用跟踪模式,如果一个对象没有任何引用指向它,那么这个对象就被认为是垃圾对象,并且可以被 ...

  5. Java垃圾回收

    垃圾收集算法 引用计数 堆中的每个对象都有一个引用计数,当对象被引用时引用计数加1,当对象的引用被重新赋值或超出有效区域时引用计数减1,当一个对象被回收后,它所引用的对象的引用计算减1.当一个对象的引 ...

  6. JavaScript具有自动垃圾回收机制

    JavaScript具有自动垃圾回收机制 原理: 找出那些不再继续使用的变量,然后释放其占用的内存.   正常的生命周期:     局部变量指在函数执行的过程中存在.而在这个过程中,会为局部变量在栈或 ...

  7. .NET 垃圾回收与内存泄漏

    > 前言相信大家一定听过,看过甚至遇到过内存泄漏.在 .NET 平台也一定知道有垃圾回收器,它可以让开发人员不必担心内存的释放问题,因为它会自定管理内存.但是在 .NET 平台下进行编程,绝对不 ...

  8. JavaScript 垃圾回收

    在公司经常会听到大牛们讨论时说道内存泄露神马的,每每都惊羡不已,最近精力主要用在了Web 开发上,读了一下<JavaScript高级程序设计>(书名很唬人,实际作者写的特别好,由浅入深)了 ...

  9. C#技术漫谈之垃圾回收机制(GC)

    GC的前世与今生 虽然本文是以.NET作为目标来讲述GC,但是GC的概念并非才诞生不久.早在1958年,由鼎鼎大名的图林奖得主John McCarthy所实现的Lisp语言就已经提供了GC的功能,这是 ...

  10. Java的垃圾回收和内存分配策略

    本文是<深入理解Java虚拟机 JVM高级特性与最佳实践>的读书笔记 在介绍Java的垃圾回收方法之前,我们先来了解一下Java虚拟机在执行Java程序的过程中把它管理的内存划分为若干个不 ...

随机推荐

  1. spring mvc 自定义Handlermapping

    上次大概写了个可以解决velocity 多视图的东西. 但是实际运用过程中又到处找了些资料看了下.这里 小计下: DispatcherServlet解析过程: ..1..HandlerMapping. ...

  2. memcpy函数的使用方法

    c和c++使用的内存拷贝函数,memcpy函数的功能是从源src所指的内存地址的起始位置开始拷贝n个字节到目标dest所指的内存地址的起始位置中. 1.函数原型 void *memcpy(void * ...

  3. 利用Asio搭建日志系统

    Asio(http://think-async.com)官方示例中给出了一个比较初步的日志服务,主要代码在basic_logger.hpp.logger_service.hpp.logger_serv ...

  4. C获取本地时间的localtime函数

    最近有朋友问如下问题: #include <stdio.h>#include <stdlib.h>#include <iconv.h>#include <ti ...

  5. 【转】探讨android更新UI的几种方法----不错

    原文网址:http://www.cnblogs.com/wenjiang/p/3180324.html 作为IT新手,总以为只要有时间,有精力,什么东西都能做出来.这种念头我也有过,但很快就熄灭了,因 ...

  6. 帕累托分析法(Pareto Analysis)(柏拉图分析)

    帕累托分析法(Pareto Analysis)(柏拉图分析) ABC分类法是由意大利经济学家帕雷托首创的.1879年,帕累托研究个人收入的分布状态图是地,发现少数人收入占全部人口收入的大部分,而多数人 ...

  7. Word Break II——LeetCode

    Given a string s and a dictionary of words dict, add spaces in s to construct a sentence where each ...

  8. 本地apt

    Ubuntu建立本地源非常實用,很多服務器在局域網沒有網络或者網络很慢的情況下,或者需要批量安裝同样的軟件的時候,如果每一台服務器都去外網下載,是不是很慢,而且也不是一個運維工程師願意這麼幹的!那有什 ...

  9. Hive从概念到安装使用总结

    一.Hive的基本概念 1.1 hive是什么? (1)Hive是建立在hadoop数据仓库基础之上的一个基础架构: (2)相当于hadoop之上的一个客户端,可以用来存储.查询和分析存储在hadoo ...

  10. SWIFT国际资金清算系统

    SWIFT又称:“环球同业银行金融电讯协会”,是国际银行同业间的国际合作组织,成立于一九七三年,目前全球大多数国家大多数银行已使用SWIFT系统.SWIFT的使用,使银行的结算提供了安全.可靠.快捷. ...