时间:2021-05-19
前言
c# 托管和非托管比较重要,因为这涉及到资源的释放。
现在只要在计算机上运行的,无论玩出什么花来,整个什么概念,逃不过输入数据修改数据输出数据(计算机本质),这里面有个数据的输入,那么我们的内存有限啊,这里面就牵扯到数据释放。
看下c# 的垃圾回收是怎么样的。
了解垃圾回收之前首先要了解数据,了解数据需要了解数据类型啊,数据类型分为值类型还有引用类型。
windows 使用一个虚拟寻址系统,该系统把程序可用的内存地址映射到硬件内存中的实际地址上,这些任务完全由windows 在后台管理。我们的程序运行在操作系统上,那么我们作为程序员关系的就是这个虚拟寻址系统。
这东西有什么用呢?
比如32位系统中,每个进程所占用的最多4G(4G这样来的,2^32,4个字节),那么这个程序如果进行管理的这4G,它不需要知道在硬件地址是多少。
比如这个进程申请了1k内存,那么这个进程管理的实际是从0到1k的虚拟内存,而不需要知道这个硬件物理内存地址是多少,有一个可以直接证明的就是我们写c++输出指针的时候,发现指针输出1千多,
你觉得可能是物理内存地址的1千多吗?默默的打开资源管理看看现在占用多少内存。
默认情况下,32 位计算机上的每个进程都具有 2 GB 的用户模式虚拟地址空间。这里解释一下,每个进程2个G是虚拟地址,就是在这个进程维护一个2G的虚拟地址,并不是实际占有2G的硬件内存地址。
盗一张图:
虚拟地址有三种状态:
状态 描述 Free 该内存块没有引用关系,可用于分配。 保留 内存块可供你使用,并且不能用于任何其他分配请求。 但是,在该内存块提交之前,你无法将数据存储到其中。 已提交 内存块已指派给物理存储。
那么这个虚拟内存上又分了堆和栈,栈上存储值类型,堆上存储引用类型。
他们的存储方式不一样。
下面是栈:
栈是这样子的先用高位后用低为,比如申请80000,先用的就是80000 直到为0为止。
{ int a=10; double b=100.0;}如上图,80000用完了,这时候栈指针指向80000。
现在int a了,int是4个字节,这时候栈指针减4,到79996这个位置。
然后是double,double 为8个字节,这时候栈指针减8,以此类推。
然后如果变量超出作用域,那么这个时候就会被垃圾回收,栈指针增加8,然后增加4。(记得栈指针增加的时候[垃圾回收]并不会去把已经使用的地址重置为0,只有类型申明的时候才重置为0,然后再赋值)
下面是引用类型:
堆是这样子的,已用的内存地址小,空闲的内存地址大。
举个栗子:
{ student a; a=new student;}首先运行student a,这个时候存储的是引用地址,也就是4个字节,存放在栈上。
然后运行a=new student(),首先假设student使用64个字节,那么在堆上就申请连续的64个字节,然后把首地址传递赋值给a。
上面非常大的程度简化了程序的内存分配,引用类型垃圾回收的一个简化版就是——当一个引用变量超出作用域的时候,它会从栈中删除,但是引用对象的数据保存在堆中,只有没有任何一个引用变量指向引用对象的时候那么这个引用对象才会回收。
好的,知道了数据存储是怎么样的,来看一下垃圾回收吧。
正文
关于垃圾回收,.net 文档是这样介绍的。
.NET 的垃圾回收器管理应用程序的内存分配和释放。
每当有对象新建时,公共语言运行时都会从托管堆为对象分配内存。
只要托管堆中有地址空间,运行时就会继续为新对象分配空间。
不过,内存并不是无限的。 垃圾回收器最终必须执行垃圾回收来释放一些内存。
垃圾回收器的优化引擎会根据所执行的分配来确定执行回收的最佳时机。
执行回收时,垃圾回收器会在托管堆中检查应用程序不再使用的对象,然后执行必要的操作来回收其内存。
那么什么时候垃圾回收呢?
1.系统具有低的物理内存。 这是通过 OS 的内存不足通知或主机指示的内存不足检测出来。
2.由托管堆上已分配的对象使用的内存超出了可接受的阈值。 随着进程的运行,此阈值会不断地进行调整。
3.调用 GC.Collect 方法。 几乎在所有情况下,你都不必调用此方法,因为垃圾回收器会持续运行。 此方法主要用于特殊情况和测试。
第三个.net 平台会帮我们处理,第二个托管堆会帮我们自我调整,关键是第1个如何物理内存不足的时候就会被回收,一般时候整个操作系统内存占用在15%-30%之间都是可调控的,基本不用担心这个问题,但是为了容灾性代码依然需要做一些判断处理。
垃圾回收的回收机制是通过代数来实现。
为优化垃圾回收器的性能,将托管堆分为三代:第 0 代、第 1 代和第 2 代,因此它可以单独处理长生存期和短生存期对象。
第 0 代。 这是最年轻的代,其中包含短生存期对象。 短生存期对象的一个示例是临时变量。 垃圾回收最常发生在此代中。
第 1 代。 这一代包含短生存期对象并用作短生存期对象和长生存期对象之间的缓冲区。
第 2 代。 这一代包含长生存期对象。 长生存期对象的一个示例是服务器应用程序中的一个包含在进程期间处于活动状态的静态数据的对象。
垃圾回收中未回收的对象也称为幸存者,并会被提升到下一代:
第 0 代垃圾回收中未被回收的对象将会升级至第 1 代。
第 1 代垃圾回收中未被回收的对象将会升级至第 2 代。
第 2 代垃圾回收中未被回收的对象将仍保留在第 2 代。
因为第 0 代和第 1 代中的对象的生存期较短,因此,这些代被称为“暂时代”。
垃圾回收的过程:
标记阶段,找到并创建所有活动对象的列表。
重定位阶段,用于更新对将要压缩的对象的引用。
压缩阶段,用于回收由死对象占用的空间,并压缩幸存的对象。 压缩阶段将垃圾回收中幸存下来的对象移至段中时间较早的一端。
因为第 2 代回收可以占用多个段,所以可以将已提升到第 2 代中的对象移动到时间较早的段中。 可以将第 1 代幸存者和第 2 代幸存者都移动到不同的段,因为它们已被提升到第 2 代。
然后后台垃圾回收、大型对象堆、被动回收、延迟模式等可以作为了解。
与我们写代码息息相关
垃圾回收机制与我们写代码息息相关的部分是:
后续一节整理一下。
以上就是带你复习c# 托管和非托管资源的详细内容,更多关于c# 托管和非托管资源的资料请关注其它相关文章!
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
IDispose模式在C++中用的很多,用来清理资源,而在C#里,资源分为托管和非托管两种,托管资源是由C#的CLR帮助我们清理的,它是通过调用对象的析构函数完
C#的每一个类型都代表一种资源,而资源又分为两类:托管资源由CLR管理分配和释放的资源,即从CLR里new出来的对象。非托管资源不受CLR管理的对象,如Wi
C#中如何合理的释放非托管内存?在本文中我们将讲解使用IDisposable释放托管内存和非托管内存。 A.首先需要让类实现IDisposable接口,然
一、托管代码/非托管代码C#代码通过C#编译器编译成程序集,程序集由微软中间语言组成,CLR会为程序集开辟一个应用程序域,程序集就是运行在这个应用程序域里面的,
C#实现IDispose接口.net的GC机制有两个问题:首先GC并不能释放所有资源,它更不能释放非托管资源。其次,GC也不是实时的,所有GC存在不确定性。为了