视频网站管理系统,可植入代码网站开发,wordpress博客vieu,seo是什么意思啊CLR垃圾回收器根据所占空间大小划分对象。大对象和小对象的处理方式有很大区别。比如内存碎片整理 —— 在内存中移动大对象的成本是昂贵的#xff0c;让我们研究一下垃圾回收器是如何处理大对象的#xff0c;大对象对程序性能有哪些潜在的影响。 大对象堆和垃圾回收 在.Net …CLR垃圾回收器根据所占空间大小划分对象。大对象和小对象的处理方式有很大区别。比如内存碎片整理 —— 在内存中移动大对象的成本是昂贵的让我们研究一下垃圾回收器是如何处理大对象的大对象对程序性能有哪些潜在的影响。 大对象堆和垃圾回收 在.Net 1.0和2.0中如果一个对象的大小超过85000byte就认为这是一个大对象。这个数字是根据性能优化的经验得到的。当一个对象申请内存大小达到这个阀值它就会被分配到大对象堆上。这意味着什么呢要理解这个我们需要理解.Net垃圾回收机制。 如大多人所知道的.Net GC是按照“代”来回收的。程序中的对象共有3代0代、1代和2代0代是最年轻的对象2代对象存活的时间最长。GC按代回收垃圾也是出于性能考虑的通常的对象都会在0代是被回收。例如在一个asp.net程序中和每一个请求相关的对象都应该在请求结束时回收掉。而没有被回收的对象会成为1代对象也就是说1代对象是常驻内存对象和马上消亡对象之间的一个缓冲区。 从代的角度看大对象属于2代对象因为只有在2代回收时才会处理大对象。当某代垃圾回收执行时会同时执行更年轻代的垃圾回收。比如当1代垃圾回收时会同时回收1代和0代的对象当2代垃圾回收时会执行1代和0代的回收. 代是垃圾回收器区分内存区域的逻辑视图。从物理存储角度看对象分配在不同的托管堆上。一个托管堆managed heap是垃圾回收器从操作系统申请的内存区通过调用windows api VirtualAlloc。当CLR载入内存之后会初始化两个托管堆一个大对象堆LOH –large object heap和一个小对象对SOH – small object heap。 内存分配请求就是将托管对象放到对应的托管堆上。如果对象的大小小于85000byte它会被放置在SOH否则会被放在LOH上。 对于SOH对象在执行一次垃圾回收之后会进入到下一代。也就是说如果在第一次执行垃圾回收时存活下来的对象会进入第二代如果在第2次垃圾回收之后该对象仍然没有被当作垃圾回收掉它就会成为2代对象2代对象就是最老的对象不会在提升代数。 当触发垃圾回收时垃圾回收器会在小对象堆做碎片整理将存活下来的对象移动到一起。而对于大对象堆由于移动内存的开销很大CLR团队选择只是清除它们将回收掉的对象组成一个列表以便满足下次有大对象申请使用内存相邻的垃圾对象会被合并成一块空闲的内存块。 需要时时留意的是直到.Net 4.0中也不会对大对象堆做碎片整理操作将来也许会做。因此如果你要分配大对象并不想他们被移动你可以使用fixed语句。 如下小对象堆SOH的回收示意图 上图中第一次垃圾回收之前有四个对象obj0-3在第一垃圾回收之后obj1和obj3被回收了同时obj2和obj0移动到一起了在第二次垃圾回收之前有分配了三个对象obj4-6在第二次执行垃圾回收之后obj2和obj5被回收了obj4和obj6被移动到obj0旁边。 下图是大对象堆LOH回收示意图 可以看到在未执行垃圾回收之前一共有四个对象obj0-3第一次二代垃圾回收之后obj1和obj2被回收掉了回收掉之后obj1和obj2所占空间被合并到了一起在obj4申请分配内存时就把obj1和obj2回收后释放的空间分配给它了同时留下了一块内存碎片。如果这个碎片的大小小于85000byte那么这个碎片就在这个程序的生命周期中永远不能被再次利用了。 如果大对象堆上没有足够的空闲内存容纳要申请的大对象空间CLR首先会尝试向操作系统申请内存如果申请失败就会触发一次二代回收来尝试释放一些内存。 在2代垃圾回收时可以将不需要的内存通过VirtualFree交还给操作系统。交还的过程参见下图 什么时候回收大对象呢 在讨论什么时候回收大对象之前先来看下普通的垃圾回收操作什么时机执行吧。垃圾回收在下列情况下发生 1. 申请的空间超过0代内存大小或者大对象堆的阀值多数的托管堆垃圾回收在这种情况下发生 2. 在程序代码中调用GC.Collect方法时如果在调用GC.Collect方法是传入GC.MaxGeneration参数时会执行所有代对象的垃圾回收包括大对象堆的垃圾回收 3. 操作系统内存不足时当应用程序收到操作系统发出的高内存通知时 4. 如果垃圾回收算法认为做二代回收是有收效时会触发二代垃圾回收 5. 每一代对象堆的都有一个所占空间大小阀值的属性当你分配对象到某一代你增长了内存总量接近了该代的阀值或者分配对象导致这一代的堆大小超过了堆阀值就会发生一次垃圾回收。因此当你分配小对象或者大对象时会对应消耗0代堆或者大对象堆的阀值。当垃圾回收器将对象代数提升到1代或者2代时会消耗1、2代的阀值。在程序运行中这些阀值是动态变化的。 大对象堆性能影响 让我们先看下分配大对象的代价。 CLR为每个新对象分配内存时都要保证这些内存清空的是没有被其他对象使用的I give out is cleared。这就意味着分配的代价完全被清理clearing的代价控制着除非在分配时触发了一次垃圾回收。如果清空1byte需要2个周期cycles就意味着清除一个最小的大对象需要170,000个周期。通常情况下人们不会分配超大的对象比如说在2GHz的机器上分配16M大小的对象大约需要16ms来清空内存。这代价太大了。 让我们在看下回收的代价。前面提到过大对象和2代龄对象一起回收。如果大对象或者2代对象占用空间超过其阀值时就会触发2代对象的回收。如果2代回收因为大对象堆超过阀值被触发2代对象堆本身没有多少对象可以做回收。如果在2代堆上没有多少对象这问题不大。但是如果2代堆很大对象很多过多的2代回收就会导致性能问题。如果是临时性的分配大对象就需要很多的时间来运行垃圾回收也就是说如果你持续的使用大对象然后又释放大对象对性能会有很大的负面影响。 大对象堆上的巨大对象通常是数组很少有一个对象很大的情况。如果对象中的元素是强引用代价会很高如果元素之间没有相互引用垃圾回收时就不需要遍历整个数组。例如用一个数组来保存二叉树的节点一种方法是在节点中强引用左右节点 ? 1 2 3 4 5 6 7 8 class Node { Data d; Node left; Node right; } Node[] binaryTree new Node[num_nodes]; 如果num_nodes是一个很大的数字就意味着每个节点都至少需要查看二个引用元素。一种替代方案是在节点中保存左右节点元素的数组索引号 ? 1 2 3 4 5 6 class Node { Data d; uint left_index; uint right_index; } 这样的话元素之间的引用关系去掉了可以通过binaryTree[left_index]来获得引用的节点。垃圾回收器在做垃圾回收时也不需要看相关的引用元素了。 为大对象堆收集性能数据 有几种方法可以收集大对象堆相关的性能数据。在我解释这些方法之前让我们先谈一下为什么需要收集大对象堆相关的性能数据。 在你开始上搜集某个方面的性能数据时有可能你已经找到这方面造成性能瓶颈的证据或者你已经没有找遍了所有方面都没有发现问题。 在查找性能问题时.Net CLR Memory 性能计数器通常是应该先考虑使用的工具。和LOH相关的计数器有generation 2 collectioins2代堆收集次数和large object heap size大对象堆大小。Generation 2 collections显示的是进程启动之后2代垃圾回收操作发生的次数。Large object heap size计数器显示的是当前大对象堆的大小值包括空闲空间这个计数器是在每次垃圾回收操作之后做更新并非每次分配内存都做更新。 可以参考下图在windows性能计数器中观察.Net CLR Memory相关性能数据 你也可以通过程序查询这些计数器的值很多人通过程序的方式收集性能计数器来帮助查找性能瓶颈。 当然也可以使用调试器winddbg观察大对象堆。 最后提示一下到目前为止大对象堆作为垃圾回收的一部分是不做内存碎片整理的但是这个只是一个clr的实现细节程序代码不应该依赖这个特点。如果要确保对象不会被垃圾回收器移动就要使用fixed语句。转载于:https://www.cnblogs.com/One-dream-man/archive/2012/12/18/2824217.html