前言

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 代。

然后后台垃圾回收、大型对象堆、被动回收、延迟模式等可以作为了解。

与我们写代码息息相关

垃圾回收机制与我们写代码息息相关的部分是:

  1. 强引用和弱引用
  2. 针对共享 Web 承载优化
  3. 垃圾回收和性能
  4. 应用程序域资源监视

后续一节整理一下。

重学c#系列——c# 托管和非托管资源(三)的更多相关文章

  1. 重学c#系列——字典(十一)

    前言 重学c#系列继续更新,简单看一下字典的源码. 看源码主要是解释一下江湖中的两个传言: 字典foreach 顺序是字典添加的顺序 字典删除元素后,字典顺序将会改变 正文 那么就从实例化开始看起,这 ...

  2. 有关 Azure IaaS VM 磁盘以及托管和非托管高级磁盘的常见问题解答

    本文将对有关 Azure 托管磁盘和 Azure 高级存储的一些常见问题进行解答. 托管磁盘 什么是 Azure 托管磁盘? 托管磁盘是一种通过处理存储帐户管理来简化 Azure IaaS VM 的磁 ...

  3. [.net 面向对象程序设计进阶] (8) 托管与非托管

    本节导读:虽然在.NET编程过程中,绝大多数内存垃圾回收由CLR(公共语言运行时)自动回收,但也有很多需要我们编码回收.掌握托管与非托管的基本知识,可以有效避免某些情况下导致的程序异常. 1.什么是托 ...

  4. NET的堆和栈04,对托管和非托管资源的垃圾回收以及内存分配

    在" .NET的堆和栈01,基本概念.值类型内存分配"中,了解了"堆"和"栈"的基本概念,以及值类型的内存分配.我们知道:当执行一个方法的时 ...

  5. C# 托管和非托管混合编程

    在非托管模块中实现你比较重要的算法,然后通过 CLR 的平台互操作,来使托管代码调用它,这样程序仍然能够正常工作,但对非托管的本地代码进行反编译,就很困难.   最直接的实现托管与非托管编程的方法就是 ...

  6. C# using 三种使用方式 C#中托管与非托管 C#托管资源和非托管资源区别

    1.using指令.using + 命名空间名字,这样可以在程序中直接用命令空间中的类型,而不必指定类型的详细命名空间,类似于Java的import,这个功能也是最常用的,几乎每个cs的程序都会用到. ...

  7. 利用C#Marshal类实现托管和非托管的相互转换

    Marshal 类 命名空间:System.Runtime.InteropServices 提供了一个方法集,这些方法用于分配非托管内存.复制非托管内存块.将托管类型转换为非托管类型,此外还提供了在与 ...

  8. [转]C# 之DLL调用(托管与非托管)

    每种编程语言调用DLL的方法都不尽相同,在此只对用C#调用DLL的方法进行介绍.首先,您需要了解什么是托管,什么是非托管.一般可以认为:非托管代码主要是基于win 32平台开发的DLL,activeX ...

  9. C#的托管与非托管大难点

    托管代码与非托管代码 众所周知,我们正常编程所用的高级语言,是无法被计算机识别的.需要先将高级语言翻译为机器语言,才能被机器理解和运行.在标准C/C++中,编译过程是这样的:源代码首先经过预处理器,对 ...

随机推荐

  1. WeChair——团队展示

    这个作业属于哪个课程 软件工程 (福州大学至诚学院 - 计算机工程系) 团队名称 WeChair 这个作业要求在哪里 团队作业第一次 这个作业的目标 团队合作,项目开发 作业正文 如下 其他参考文献 ...

  2. Asp.Net 五大对象及作用

    Connection(连接对象):与数据源建立连接. DataAdapter(适配器对象):对数据源执行操作并返回结果,在DataSet与数据源之间建立通信,将数据源中的数据写入DataSet中,或根 ...

  3. JAVA 字节流 与 字符流 的区别

    字节流与和字符流的使用非常相似,两者除了操作代码上的不同之外,是否还有其他的不同呢? 字节流 在操作时本身不会用到缓冲区(内存),是文件本身直接操作的 字符流 在操作时使用了缓冲区,通过缓冲区再操作文 ...

  4. 尚硅谷 dubbo学习视频

    1 1.搭建zookpeer注册中心 windows下载zooker  需要修改下zoo_sample .cfg为zoo.cnf 然后需要zoo.cnf中数据文件的路径 第五步:把zoo_sample ...

  5. 九、深度优先 && 广度优先

    原文地址 一.什么是"搜索"算法? 算法是作用于具体数据结构之上的,深度优先搜索算法和广度优先搜索算法都是基于"图"这种数据结构的. 因为图这种数据结构的表达能 ...

  6. CImage显示位图与CDC双缓冲冲突,使用路径层解决.

    2010年04月29日 星期四 20:35 位图闪的问题困扰我很久了,因为程序的需要,我显示位图的方式是CImage类. 如果从CImage转到CBitmap,之后使用Attach到是可以,但我发现这 ...

  7. Laravel表单验证提示设置多语言

    默认表单提示是英文的,我们可以安装语言包构建多语言环境. 根据版本选择命令 For Laravel 7.x : run composer require caouecs/laravel-lang:~6 ...

  8. Tomcat的启停脚本源码解析

    Tomcat是一款我们平时开发过程中最常用到的Servlet容器.本系列博客会记录Tomcat的整体架构.主要组件.IO线程模型.请求在Tomcat内部的流转过程以及一些Tomcat调优的相关知识. ...

  9. 请解释ASP. NET中的web页面与隐藏类之间的关系

    请解释ASP.NET中的web页面与其隐藏类之间的关系 其实页面与其隐藏类之间就是一个部分类的关系,你在页面上放一个一个的控件就是在这个类中定义一个一个的属性, 因为是同一个类的部分类的关系,所以隐藏 ...

  10. js 字符串转方法,this域绑定

    闲着没事,开发一个列表页面配置的功能,其中涉及到了按钮点击事件,在页面进行编辑,保存到数据库中.写好的js脚本,function是字符串格式,所以要让生成的脚本生效,还要做一些操作. 1.首先保存在数 ...