亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C#垃圾回收的大對象

2019-11-10 20:51:15
字體:
來源:轉載
供稿:網友

轉載自:http://blog.csdn.net/u010032648/article/details/50298663

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和obj1移動到一起了;在第二次垃圾回收之前有分配了三個對象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代回收就會導致性能問題。如果是臨時性的分配大對象,就需要很多的時間來運行垃圾回收;也就是說如果你持續的使用大對象然后又釋放大對象對性能會有很大的負面影響。

大對象堆上的巨大對象通常是數組(很少有一個對象很大的情況)。如果對象中的元素是強引用,代價會很高;如果元素之間沒有相互引用,垃圾回收時就不需要遍歷整個數組。例如:用一個數組來保存二叉樹的節點,一種方法是在節點中強引用左右節點:

class Node{    Data d;    Node left;    Node right;};Node[] binary_tr = new Node [num_nodes];

如果num_nodes是一個很大的數字,就意味著每個節點都至少需要查看二個引用元素。一種替代方案是在節點中保存左右節點元素的數組索引號

class Node{    Data d;    uint left_index;    uint right_index;};

這樣的話,元素之間的引用關系去掉了;可以通過binaryTree[left_index]來獲得引用的節點。垃圾回收器在做垃圾回收時也不需要看相關的引用元素了。

為大對象堆收集性能數據

有幾種方法可以收集大對象堆相關的性能數據。在我解釋這些方法之前,讓我們先談一下為什么需要收集大對象堆相關的性能數據。

在你開始上搜集某個方面的性能數據時,有可能你已經找到這方面造成性能瓶頸的證據;或者你已經沒有找遍了所有方面都沒有發現問題。

在查找性能問題時.Net CLR Memory 性能計數器通常是應該先考慮使用的工具。和LOH相關的計數器有generation 2 collectioins(2代堆收集次數)和large object heap size大對象堆大小。Generation 2 collections顯示的是進程啟動之后2代垃圾回收操作發生的次數。Large object heap size計數器顯示的是當前大對象堆的大小值,包括空閑空間;這個計數器是在每次垃圾回收操作之后做更新,并非每次分配內存都做更新。

可以參考下圖在windows性能計數器中觀察.Net CLR Memory相關性能數據

你也可以通過程序查詢這些計數器的值;很多人通過程序的方式收集性能計數器來幫助查找性能瓶頸。

當然也可以使用調試器winddbg觀察大對象堆。

最后提示一下:到目前為止,大對象堆作為垃圾回收的一部分是不做內存碎片整理的,但是這個只是一個clr的實現細節,程序代碼不應該依賴這個特點。如果要確保對象不會被垃圾回收器移動,就要使用fixed語句。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产精品一区二区久久国产| 亚洲精品国产精品乱码不99按摩| 91免费的视频在线播放| 国产91|九色| 欧美精品在线极品| 亚洲综合最新在线| 亚洲韩国欧洲国产日产av| 欧美性色xo影院| 亚洲色图狂野欧美| 中文字幕国产精品久久| 亚洲欧美制服第一页| 国产亚洲免费的视频看| 欧美二区在线播放| 亚洲色图欧美制服丝袜另类第一页| 久久99精品久久久久久噜噜| 日韩一区二区福利| 欧美日本中文字幕| 欧美成人免费视频| 欧美有码在线观看| 久久久亚洲欧洲日产国码aⅴ| 亚洲精品网站在线播放gif| 欧美性猛交xxxx乱大交| 九九热在线精品视频| 狠狠躁天天躁日日躁欧美| 国产精品久久久久久久久久尿| 亚洲高清在线观看| 欧美激情18p| 国产偷亚洲偷欧美偷精品| 国产99在线|中文| 97视频在线看| 国产精品一区电影| 国产在线精品自拍| 日日狠狠久久偷偷四色综合免费| 亚洲成人免费在线视频| 日韩中文视频免费在线观看| 欧美日韩在线免费| 成人国产精品免费视频| 高清一区二区三区日本久| 国产在线拍揄自揄视频不卡99| 久久免费视频这里只有精品| 亚洲va国产va天堂va久久| 日韩在线播放一区| 久久久久久91| 亚洲香蕉成视频在线观看| 97色伦亚洲国产| 成人午夜两性视频| 91精品在线观| 中文字幕日韩免费视频| 在线观看国产成人av片| 日韩精品免费在线播放| 午夜剧场成人观在线视频免费观看| 欧美激情一区二区三区高清视频| 国语自产在线不卡| 欧美在线视频在线播放完整版免费观看| 国产精品女主播视频| 国产精品99久久久久久久久| 亚洲精品永久免费精品| 欧美激情免费观看| 欧美黄色免费网站| 在线视频精品一| 欧美精品一二区| 黑人巨大精品欧美一区免费视频| 中文字幕精品国产| 亚洲精品自拍视频| 91精品国产色综合久久不卡98口| 精品女同一区二区三区在线播放| 日韩精品福利网站| 中文字幕日韩精品在线观看| 国产精品久久久久av| 亚洲一区av在线播放| 久久国产精品久久久久| 国产精品久久久久久久7电影| 欧美国产亚洲视频| 性欧美亚洲xxxx乳在线观看| 国产精品色午夜在线观看| 久久久久久国产精品美女| 亚洲国内精品在线| www.日韩视频| 久久久久久久国产精品视频| www.亚洲免费视频| 自拍偷拍亚洲在线| 国产精品免费一区二区三区都可以| 日韩在线视频观看正片免费网站| 欧美巨乳美女视频| 青青草成人在线| www亚洲欧美| 国产亚洲精品久久久优势| 一本一本久久a久久精品牛牛影视| 国产精品一区二区在线| 亚洲摸下面视频| 国产91免费观看| 日韩久久免费视频| 国产精品成av人在线视午夜片| 国语自产精品视频在线看抢先版图片| 45www国产精品网站| 欧美中文在线观看| 欧美电影免费观看| 日本一区二区三区在线播放| 亚洲影视中文字幕| 成人高h视频在线| 日韩动漫免费观看电视剧高清| 欧美大片第1页| 欧美性猛交xxxx免费看久久久| 国产va免费精品高清在线观看| 欧美最猛性xxxxx亚洲精品| 一本色道久久综合狠狠躁篇的优点| 亚洲黄色av网站| 国产精品白丝av嫩草影院| 精品久久久香蕉免费精品视频| 日韩中文字幕在线精品| 国产91精品高潮白浆喷水| 韩国精品美女www爽爽爽视频| 亚洲一区美女视频在线观看免费| 午夜美女久久久久爽久久| 久久免费少妇高潮久久精品99| 精品国产1区2区| 日韩美女毛茸茸| 亚洲黄在线观看| 国产精品视频精品| 九九热最新视频//这里只有精品| 亚洲成年人在线| 少妇激情综合网| 国产精品久久久久免费a∨大胸| 国产精品成熟老女人| 欧美在线影院在线视频| 精品视频中文字幕| 日韩美女视频中文字幕| 91精品啪aⅴ在线观看国产| 亚洲人免费视频| 成人国内精品久久久久一区| 国产精品久久97| 91在线无精精品一区二区| 91中文字幕在线观看| 成人网在线免费看| 亚洲人在线观看| 欧美另类69精品久久久久9999| 亚洲最大av网| 欧美另类高清videos| 欧美xxxx18性欧美| 欧美国产视频一区二区| 国产精品中文在线| 欧美激情一区二区三区在线视频观看| 尤物九九久久国产精品的特点| 精品视频一区在线视频| 日韩欧美极品在线观看| 国产美女高潮久久白浆| 日韩毛片在线看| 亚洲bt天天射| 久久影视电视剧免费网站清宫辞电视| 91夜夜揉人人捏人人添红杏| 国产精品美女网站| 久久91精品国产91久久久| 亚洲第一福利视频| 日本伊人精品一区二区三区介绍| 国产亚洲成av人片在线观看桃| 免费av在线一区| 亚洲精品在线观看www| 久久福利视频导航| 国产精品99久久久久久久久| 欧美日韩成人在线视频| 成人a免费视频| 欧美贵妇videos办公室| 欧美日韩国产精品一区| 亚洲永久免费观看|