自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

.Net垃圾收集機制 了解算法與代齡

開發(fā) 后端 算法
組成.Net平臺一個很重要的部分----垃圾收集器(Garbage Collection),今天我們就來講講它,想想看沒有GC,.Net還能稱之為一個平臺嗎?各種語言雖然都被編譯成MSIL,但是運行時的資源回收工作卻“各自為戰(zhàn)”,這樣不但增加了編程難度,也會使內(nèi)存管理工作變得復(fù)雜無比,更也不利于平臺移植。

垃圾收集器在本質(zhì)上就是負(fù)責(zé)跟蹤所有對象被引用到的地方,關(guān)注對象不再被引用的情況,回收相應(yīng)的內(nèi)存。在.NET平臺中同樣如此,有效的提高.NET垃圾回收性能,能夠提高程序執(zhí)行效率。

其實垃圾收集并不是伴隨Java出現(xiàn)的,早在1958年,圖林獎得主John發(fā)明的Lisp語言就已經(jīng)提供了GC的功能,這是GC的第一次出現(xiàn),是思想的一次閃光!而后,1984年Dave Ungar發(fā)明的Small talk語言第一次正式采用了GC機制。.Net的垃圾回收機制是個很大的話題,如果你沒接觸過類似C++那樣的語言,就很難理解GC是一個多么重要、令人興奮的東西:

1.提高軟件系統(tǒng)的內(nèi)聚。

2.降低編程復(fù)雜度,使程序員不必分散精力去處理析構(gòu)。

3.不妨礙設(shè)計師進行系統(tǒng)抽象。

4.減少由于內(nèi)存運用不當(dāng)產(chǎn)生的Bug。

5.成功的將內(nèi)存管理工作從程序的編寫時,脫離至運行時,使不可預(yù)估的管理漏洞變?yōu)榭深A(yù)估的。

1.算法

垃圾收集器的本質(zhì),就是跟蹤所有被引用到的對象,整理對象不再被引用的對象,回收相應(yīng)的內(nèi)存。這聽起來類似于一種叫做“引用計數(shù)(Reference Counting)”的算法,然而這種算法需要遍歷所有對象,并維護它們的引用情況,所以效率較低些,并且在出現(xiàn)“環(huán)引用”時很容易造成內(nèi)存泄露。所以.Net中采用了一種叫做“標(biāo)記與清除(Mark Sweep)”算法來完成上述任務(wù)。“標(biāo)記與清除”算法,顧名思義,這種算法有兩個本領(lǐng):

“標(biāo)記”本領(lǐng)——垃圾的識別:從應(yīng)用程序的root出發(fā),利用相互引用關(guān)系,遍歷其在Heap上動態(tài)分配的所有對象,沒有被引用的對象不被標(biāo)記,即成為垃圾;存活的對象被標(biāo)記,即維護成了一張“根-對象可達圖”。其實,CLR會把對象關(guān)系看做“樹圖”,無疑,了解數(shù)據(jù)結(jié)構(gòu)的同學(xué)都知道,有了“樹圖”的概念,會加快遍歷對象的速度。

檢測、標(biāo)記對象引用,是一件很有意思的事情,有很多方法可以做到,但是只有一種是效率最優(yōu)的,.Net中是利用棧來完成的,在不斷的入棧與出棧中完成檢測:先在樹圖中選擇一個需要檢測的對象,將該對象的所有引用壓棧,如此反復(fù)直到棧變空為止。棧變空意味著已經(jīng)遍歷了這個局部根(或者說是樹圖中的節(jié)點)能夠到達的所有對象。樹圖節(jié)點范圍包括局部變量(實際上局部變量會很快被回收,因為它的作用域很明顯、很好控制)、寄存器、靜態(tài)變量,這些元素都要重復(fù)這個操作。一旦完成,便逐個對象地檢查內(nèi)存,沒有標(biāo)記的對象變成了垃圾。

“清除”本領(lǐng)——回收內(nèi)存:啟用Compact算法,對內(nèi)存中存活的對象進行移動,修改它們的指針,使之在內(nèi)存中連續(xù),這樣空閑的內(nèi)存也就連續(xù)了,這就解決了內(nèi)存碎片問題,當(dāng)再次為新對象分配內(nèi)存時,CLR不必在充滿碎片的內(nèi)存中尋找適合新對象的內(nèi)存空間,所以分配速度會大大提高。

但是大對象(large object heap)除外,GC不會移動一個內(nèi)存中巨無霸,因為它知道現(xiàn)在的CPU不便宜。通常,大對象具有很長的生存期,當(dāng)一個大對象在.NET托管堆中產(chǎn)生時,它被分配在堆的一個特殊部分中,移動大對象所帶來的開銷超過了整理這部分堆所能提高的性能。

Compact算法除了會提高再次分配內(nèi)存的速度,如果新分配的對象在堆中位置很緊湊的話,高速緩存的性能將會得到提高,因為一起分配的對象經(jīng)常被一起使用(程序的局部性原理),所以為程序提供一段連續(xù)空白的內(nèi)存空間是很重要的。 #p#

2.代齡(Generation)

代齡就是對Heap中的對象按照存在時間長短進行分代,最短的分在第0代,最長的分在第2代,第2代中的對象往往是比較大的。Generation的層級與FrameWork版本有關(guān),可以通過調(diào)用GC.MaxGeneration得知。

通常,GC會優(yōu)先收集那些最近分配的對象(第0代),這與操作系統(tǒng)經(jīng)典內(nèi)存換頁算法“最近最少使用”算法如出一轍。但是,這并不代表GC只收集最近分配的對象,通常,.Net GC將堆空間按對象的生存期長短分成3代:新分配的對象在第0代(0代空間最大長度通常為256K),按地址順序分配,它們通常是一些局部變量;第1代(1代空間最大長度通常為2 MB)是經(jīng)過0代垃圾收集后仍然駐留在內(nèi)存中的對象,它們通常是一些如表單,按鈕等對象;第2代是經(jīng)歷過幾次垃圾收集后仍然駐留在內(nèi)存中的對象,它們通常是一些應(yīng)用程序?qū)ο蟆?/p>

當(dāng)內(nèi)存吃緊時(例如0代對象充滿),GC便被調(diào)入執(zhí)行引擎——也就是CLR——開始對第0代的空間進行標(biāo)記與壓縮工作、回收工作,這通常小于1毫秒。如果回收后內(nèi)存依然吃緊,那么GC會繼續(xù)回收第1代(回收操作通常小于10毫秒)、第2代,當(dāng)然GC有時并不是按照第0、1、2代的順序收集垃圾的,這取決于運行時的情況,或是手動調(diào)用GC.Collect(i)指定回收的代。當(dāng)對第2代回收后任然無法獲得足夠的內(nèi)存,那么系統(tǒng)就會拋出OutOfMemoryException異常,當(dāng)經(jīng)過幾次GC過后,0代中的某個對象仍然存在,那么它將被移動到第1代。同理,第1、2代也按同樣的邏輯運行。

這里還要說的是,GC Heap中代的數(shù)量與容量,都是可變的(這由一個“策略引擎”控制,在第二節(jié)中,會介紹到“策略引擎”), 以下代碼結(jié)合Windbg可以說明這個問題,以下代碼中,可以通過單擊按鈕“button1”,不斷的分配內(nèi)存,而后獲得對象“a”的代齡情況,并且在Form加載時也會獲得“a”的代齡。

  1. public partial class Form1 : Form  
  2. {  
  3.         private string a = new string('a',1);  
  4.         public Form1()  
  5.         {  
  6.             InitializeComponent();  
  7.         }  
  8.         private void button1_Click(object sender, EventArgs e)  
  9.         {  
  10.             a = new string('a', 900000);  
  11.             label1.Text = GC.GetGeneration(a).ToString();  
  12.         }  
  13.         private void Form1_Load(object sender, EventArgs e)  
  14.         {  
  15.             label1.Text = GC.GetGeneration(a).ToString();  
  16.         }  

程序剛加載時,“a”的代齡為第0代,通過windbg我們還獲得了以下信息:

windbg信息

可以看出,GC堆被分成了兩個段,三代,每代起始地址十進制差值為12,點擊數(shù)次“button1”按鈕后,“a”的代齡升為第2代,通過windbg我們又獲得了以下信息:

windbg信息

這里要注意一個很關(guān)鍵的地方,就是各代的起始(generation x starts at)十進制地址差值不再是12,0代與1代差為98904,1代與2代差為107908,這說明代的大小隨程序運行在改變,并且GC heap的大小也有變化。

文章原標(biāo)題:.Net Discovery系列之三--深入理解.Net垃圾收集機制(上)

原文鏈接:http://www.cnblogs.com/isline/archive/2009/03/03/1402350.html

【編輯推薦】

  1. 淺析提高.NET垃圾回收性能的幾種方法
  2. .Net Framework垃圾收集具體算法詳解
  3. .NET Framework回收內(nèi)存操作細節(jié)披露
  4. .NET Framework 4.0功能特點詳細講解
  5. 深入.NET Framework 4.0 關(guān)于Lazy的點滴
責(zé)任編輯:王曉東 來源: 博客園
相關(guān)推薦

2024-07-15 08:00:00

2010-01-06 16:33:50

.Net Framew

2009-10-30 10:47:48

VB.NET垃圾收集器

2021-11-05 15:23:20

JVM回收算法

2009-06-15 16:14:40

Java垃圾收集算法GC

2010-03-04 14:33:11

.NET垃圾收集

2009-09-02 09:23:26

.NET內(nèi)存管理機制

2024-01-15 11:12:28

Go內(nèi)存開發(fā)

2024-05-28 00:00:03

Java垃圾收集機制

2010-01-14 11:28:54

JVM分代垃圾回收

2017-09-21 14:40:06

jvm算法收集器

2024-03-15 08:04:30

G1CMSJVM

2023-02-26 11:50:04

Hbase程序Oracle

2011-12-26 09:50:05

.NET垃圾回收

2020-10-26 13:42:28

Python算法垃圾

2022-05-06 22:13:56

JVM垃圾收集算法

2010-09-26 13:29:46

JVM垃圾回收

2017-08-04 10:53:30

回收算法JVM垃圾回收器

2023-06-09 08:11:32

2020-05-14 13:39:19

Java 垃圾回收機制
點贊
收藏

51CTO技術(shù)棧公眾號