C#垃圾回收和析构函数以及弱引用

2024-06-07 17:48

本文主要是介绍C#垃圾回收和析构函数以及弱引用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在程序中,当我们每创建一个对象,就会在内存中开辟一个空间,用以存放这个对象。如果创建的对象多了,内存就会出现不够用的情况。这时我们就要把内存中不再使用的对象释放掉,避免内存的占用及程序的异常。这个过程就是垃圾回收。手动进行垃圾回收的方法是:GC.Collect(); ,但我们一般不会这么做,一来是因为我们声明的对象有可能还会被引用,而且在手动回收的时候会漏掉很多,二来VS为每个程序提供自动回收垃圾的功能,不需要我们手动进行。

CLR的一个核心功能就是进行垃圾回收。垃圾回收的目的就是提高内存利用率。

垃圾回收器,只回收托管堆中的内存资源,不回收其它资源(数据库连接、文件句柄、网络端口等)。

那什么样的对象才会被回收呢?在内存中,只要没有变量引用的对象,表示这个对象可以被回收了(null)。但是执行垃圾回收的时间是不确定的,一般是当程序需要新内存的时候开始执行回收。另外我们还可以手动调用垃圾回收器:GC.Collect(); ,但不建议使用,因为程序流程在垃圾回收的时候会短暂的暂停一下。

在C#中,垃圾回收器是用“代”的概念来区分的。一共有3代:第0代,第一代,第二代。各代的回收机率是第0代最高,第一代次之,最后是第2代。它的执行过程是当第0代进行垃圾回收之后,剩余没有被释放的对象就会转到第一代里面去。如果第1代的内存已满,再来把第1代中的对象进行垃圾回收,把剩余没被释放的对象存入第2代,这样第1代就会有空间存放第0代中未被释放的对象。对于第2代,如果内存已满,也会进行内存释放,如果这些对象都在使用而无法释放,就会报错说内存空间不够用。从上面可以看出,越老的对象生存机率越大。

前面已经提到,垃圾回收只回收托管堆中的内存资源,不回收其它资源。那其它资源该如何回收呢?这时就可以使用析构函数或Dispose()方法。

类的析构函数由带有~前缀的类名(与构造函数的相同)来声明。在.Net中使用的析构函数(由System.Object提供)叫做Finalize(),但这不是我们用于声明析构函数的名称。如下所示:

class MyClass

{

~MyClass()

{

//Destructor body

}

}

 当进行垃圾回收时,就执行析构函数中的代码,释放资源。在调用这个析构函数之后,还将隐式地调用基类的析构函数,包括System.Object根类中的Finalize()调用。这个技术可以让.NET Framework确保调用Finalize(),因为重写Finalize()的指基类调用需要显式的执行,这是有潜在危险的。上面的代码在C++中叫析构函数,在C#中叫Finalize()函数,在当前对象被垃圾回收之前会调用这个函数,释放其它资源。但是我们不能手动调用析构函数,所以一般回收其它内存,都写在Dispose()方法中。而Dispose()方法由IDisposeable接口提供,因此重写Dispose()方法的类需要继承这个接口。由于Dispose()方法已经可以释放其它资源,这时就不需要再调用Finalize()方法了,所以在Dispose()方法里面通过GC.SuppressFinalize(this)方法告诉程序不用调用Finalize()方法了。

再说关于弱引用,先看下面的代码:

class Program{static void Main(string[] args){Person p = new Person();p.Age = 18;p = null;Console.ReadKey();}}public class Person{public int Age { get; set; }}
在上面的代码中,我们先创建了一个Person类的对象,然后给它赋值。当程序执行到p=null的时候,如果再想用p指向程序开始创建的对象(new Person()),是不可能的,因为这个对象虽然存在于内存中,但没有任何引用指向它,无法找到它的地址。如果想要得到这个对象,可以使用弱引用。如下所示:
static void Main(string[] args){            Person p = new Person();  //创建一个对象p.Age = 18;WeakReference wr = new WeakReference(p);  //把这个对象传给弱引用p = null;     //对这个对象进行垃圾回收object pnew = wr.Target;   //获取弱引用中刚才创建的对象if (pnew != null){Console.WriteLine(((Person)pnew).Age);}else{Console.WriteLine("对象已经被回收!");}Console.ReadKey();}
运行程序,显示的字符串是18。使用弱引用需要创建一个WeakReference对象,并且把要进行回收的对象传过去,这样在垃圾回收之后,我们仍然可以得到刚才创建的对象。这与直接让另外一个Person类型的变量p1来指向new Person()不同,用p1指向new Person()不能对象这个对象进行回收,而使用弱引用却可以把这个对象回收,释放资源。

如下所示,我们在上面的代码中加一个回收器:

static void Main(string[] args){            Person p = new Person();  //创建一个对象p.Age = 18;WeakReference wr = new WeakReference(p);  //把这个对象传给弱引用p = null;     //对这个对象进行垃圾回收GC.Collect();   //手动调用系统的垃圾回收器object pnew = wr.Target;   //获取弱引用中刚才创建的对象if (pnew != null){Console.WriteLine(((Person)pnew).Age);}else{Console.WriteLine("对象已经被回收!");}Console.ReadKey();}
再次运行程序,我们会发现这个对象已经不存在了,因为它已经被当作垃圾回收处理。弱引用一般用于那些创建时比较耗时的对象。





这篇关于C#垃圾回收和析构函数以及弱引用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1039875

相关文章

Python中help()和dir()函数的使用

《Python中help()和dir()函数的使用》我们经常需要查看某个对象(如模块、类、函数等)的属性和方法,Python提供了两个内置函数help()和dir(),它们可以帮助我们快速了解代... 目录1. 引言2. help() 函数2.1 作用2.2 使用方法2.3 示例(1) 查看内置函数的帮助(

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式

《C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式》Markdown凭借简洁的语法、优良的可读性,以及对版本控制系统的高度兼容性,逐渐成为最受欢迎的文档格式... 目录为什么要将文档转换为 Markdown 格式使用工具将 Word 文档转换为 Markdown(.

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

Python中bisect_left 函数实现高效插入与有序列表管理

《Python中bisect_left函数实现高效插入与有序列表管理》Python的bisect_left函数通过二分查找高效定位有序列表插入位置,与bisect_right的区别在于处理重复元素时... 目录一、bisect_left 基本介绍1.1 函数定义1.2 核心功能二、bisect_left 与

java中BigDecimal里面的subtract函数介绍及实现方法

《java中BigDecimal里面的subtract函数介绍及实现方法》在Java中实现减法操作需要根据数据类型选择不同方法,主要分为数值型减法和字符串减法两种场景,本文给大家介绍java中BigD... 目录Java中BigDecimal里面的subtract函数的意思?一、数值型减法(高精度计算)1.

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以