1.内存基础知识数组
可用。 该内存块没有引用关系,可用于分配。安全
保留。 内存块可供您使用,而且不能用于任何其余分配请求。 可是,在该内存块提交以前,您没法将数据存储到其中。服务器
提交。 内存块已指派给物理存储。并发
2.垃圾回收的条件函数
当知足如下条件之一时将发生垃圾回收:测试
3.托管堆spa
在垃圾回收器由 CLR 初始化以后,它会分配一段内存用于存储和管理对象。 此内存称为托管堆(与操做系统中的本机堆相对)。操作系统
每一个托管进程都有一个托管堆。 进程中的全部线程都在同一堆上分配对象。线程
当触发垃圾回收时,垃圾回收器将回收由死对象占用的内存。 回收进程会对活动对象进行压缩,以便将它们一块儿移动,并移除死空间,从而使堆更小一些。 这将确保一块儿分配的对象全都位于托管堆上,从而保留它们的局部性。对象
垃圾回收的侵入性(频率和持续时间)是由分配的数量和托管堆上保留的内存数量决定的。
此堆可视为两个堆的累计:大对象堆和小对象堆。
大对象堆包含其大小为 85,000 个字节和更多字节的对象。大对象堆上的特大对象一般是数组。 很是大的实例对象是不多见的。
4.代数
堆上的对象有三代:
当条件获得知足时,垃圾回收将在特定代上发生。 回收某个代意味着回收此代中的对象及其全部更年轻的代。 第 2 代垃圾回收也称为完整垃圾回收,由于它回收全部代上的全部对象(即,托管堆中的全部对象)。
幸存和提高:垃圾回收中未回收的对象也称为幸存者,并会被提高到下一代。 在第 0 代垃圾回收中幸存的对象将被提高到第 1 代;在第 1 代垃圾回收中幸存的对象将被提高到第 2 代;而在第 2 代垃圾回收中幸存的对象将仍为第 2 代。
当垃圾回收器检测到某个代中的幸存率很高时,它会增长该代的分配阈值,所以下一次回收将会获取一个很是大的回收内存。 CLR 会在如下两个优先级别以前进行平衡:不容许应用程序的工做集获取太大内存以及不容许垃圾回收花费太多时间。
5.垃圾回收过程当中发生的状况
垃圾回收分为如下几个阶段:
由于第 2 代回收能够占用多个段,因此能够将已提高到第 2 代中的对象移动到时间较早的段中。 能够将第 1 代幸存者和第 2 代幸存者都移动到不一样的段,由于它们已被提高到第 2 代。
将不会压缩大对象堆,由于这会在一个不可接受的时间长度内增长内存使用量。
垃圾回收器使用如下信息来肯定对象是否为活动对象:
在垃圾回收启动以前,除了触发垃圾回收的线程之外的全部托管线程均会挂起。
下图演示了触发垃圾回收并致使其余线程挂起的线程。
6.后台垃圾回收
在后台垃圾回收中,在进行第 2 代回收的过程当中,将会根据须要收集暂时代(第 0 代和第 1 代)。 后台垃圾回收没法设置;它会自动运行并启用并发垃圾回收。 后台垃圾回收是对并发垃圾回收的替代。 与并发垃圾回收同样,后台垃圾回收是在一个专用线程上执行的而且只适用于第 2 代回收。
后台垃圾回收期间对暂时代的回收称为前台垃圾回收。 发生前台垃圾回收时,全部托管线程都将被挂起。
当后台垃圾回收正在进行而且您已在第 0 代中分配了足够的对象时,CLR 将执行第 0 代或第 1 代前台垃圾回收。 专用的后台垃圾回收线程将在常见的安全点上进行检查以肯定是否存在对前台垃圾回收的请求。 若是存在,则后台回收将挂起自身以便前台垃圾回收能够发生。 在前台垃圾回收完成以后,专用的后台垃圾回收线程和用户线程将继续。
后台垃圾回收能够消除并发垃圾回收所带来的分配限制,由于在后台垃圾回收期间,可发生暂时垃圾回收。 这意味着,后台垃圾回收能够移除暂时代中的死对象,并且还能够在第 1 代垃圾回收期间根据须要展开堆。
后台垃圾回收当前不可用于服务器垃圾回收