在.net中内存资源可以分为托管资源和非托管资源。托管资源受CLR的管理,非托管资源则不受。 1、托管资源托管资源分别存放在“栈”和“托管堆”。 “值类型的实例作为局部变量时”和“引用类型的实例的引用”会存...
在.net中内存资源可以分为托管资源和非托管资源。托管资源受CLR的管理,非托管资源则不受。
1、托管资源
托管资源分别存放在“栈”和“托管堆”。
“值类型的实例作为局部变量时”和“引用类型的实例的引用”会存放在栈上。
“值类型的实例作为类的成员时”和“引用类型的实例”会存放在堆上。
1.1、栈
栈又称线程栈,空间比较小,是一个先进后出的结构,内存是由地址高往低填充。
栈有一个指针,指向栈的下一个可用空间,当有变量入栈时,指针往低地址移动,当变量出栈时,指针回到高地址。
public static void Main() { int n = 1; char c = 'a'; }
以上述代码为例,假设地址从2000开始。代码从Main开始执行,首先进入的是变量n,因为是整型所以占用4个字节,指针下移四个字节,变量n的值1存的地址为1997~2000。
下一行代码,为字符型变量c分配两个字节的空间,指针下移两个字节,变量c的值‘a’存的地址为1995~1996。
最后执行到Main函数的右括号,执行结束,删除num和c的值。指针先上移2个字节,删除c的值,再上移4个字节,删除num的值。
由于栈的内存是连续的,且只能从栈顶拿出或进入,所以不需要进行GC。
1.2、堆
堆是托管堆的简称,是由CLR进行管理的。它是程序运行时,CLR申请的一块内存,是属于进程内存空间的一部分。
这块空间可以划分为:
1.GC堆:分为0,1,2代三块区域,代越大空间越大,其中大对象堆(large object heap)是2代堆的一部分。
2.几个程序域:上面有各自的加载堆(loader heap),加载堆上存了方法表(Method Table),上面存储了静态字段、所有的函数、基类型、实现的接口等等。加载堆不受GC影响,其生命周期为创建到程序域卸载。
GC的三代回收机制:新创建的对象是第0代,当第0代满时,便触发垃圾回收,没有被回收的则会变成第1代。以此类推,当1代回收时,留下来的变成了2代。
大对象堆:超过85KB的实例会存放在大对象堆上。
堆中实例的布局如下:
同步块索引(synchronization block index):同步块索引,又被叫做对象头指针,位于对象地址的-4到0字节(以32位机型为例)。
同步块索引在线程同步中用来判断对象是被使用还是闲置。
当有线程要对某个对象操作时,检查其同步块索引的值,若为表示闲置的值,便进行操作。
然后在CLR的同步块数组中增加一块新的同步块,把此块的索引值写入同步块的索引值里,表示其被占用。
线程用完后,便把索引值改回表示闲置的值。
方法表指针:方法表指针指向类型对象。
类型对象是加载程序集时,在加载堆中创建的。
类型对象存储了静态字段和方法表,被所有该类型的实例共享。
方法表包括所有函数,包括静态函数和非静态函数。
2、非托管资源
非托管资源是不受CLR管理的资源,最常见的非托管资源是包装操作系统资源的对象,例如文件、网络连接、数据库连接等。
C#提供Object.Finalize函数在GC时调用来释放这类资源,这个函数默认是空的,不过Object.Finalize是无法重载的,它是根据类的析构函数自动生成的,因此要将释放非托管资源的代码写在析构函数里。
说下GC调用Finalize的流程:
new一个包含Finalize函数的对象,开辟内存后,指向它的指针会被存放到终结列表(Finalization List)。
垃圾回收时,被当作垃圾的对象如果同时存在于终结列表里,就会将该对象从终结列表中移除,并存入终结可达列表(F-reachable List)。并且这些对象变为可达,不会被GC回收,即表示这些对象提升了一代。
该队列中的对象都是可达的,并需要执行Finalize函数。执行Finalize函数是由一个高优先级的CLR线程进行的,执行完毕后,会将对象的指针从Freachable Queue中移除。
当再次进行垃圾回收时,原终结可达列表里的对象经过处理都变为不可达对象,只有当这一代内存不足时才会对对象进行垃圾回收,这些对象内存才会真正释放掉。所以含有Finalize函数的对象最少要经过两次垃圾回收才会被真正释放。
因为非托管资源比较有限,等GC调用Finalize释放,会让非托管资源处于不必要的空闲状态,影响性能。因此定义了一个IDispose接口,用于手动释放非托管资源。
在一个包含非托管资源的类中,关于资源释放的标准做法是:
① 继承IDisposable接口;
② 实现Dispose()方法,在其中释放托管资源和非托管资源,并将对象本身从垃圾回收器中移除(垃圾回收器不在回收此资源);
③ 实现类析构函数,在其中释放非托管资源。
提一下,继承了IDispose接口的类可以使用using,当超出作用域后,系统会自动调用Dispose函数。
本文标题为:c#内存管理
基础教程推荐
- C# 调用WebService的方法 2023-03-09
- C#类和结构详解 2023-05-30
- linux – 如何在Debian Jessie中安装dotnet core sdk 2023-09-26
- C#控制台实现飞行棋小游戏 2023-04-22
- C# List实现行转列的通用方案 2022-11-02
- ZooKeeper的安装及部署教程 2023-01-22
- 一个读写csv文件的C#类 2022-11-06
- winform把Office转成PDF文件 2023-06-14
- C# windows语音识别与朗读实例 2023-04-27
- unity实现动态排行榜 2023-04-27