成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

閑談.NET Framework自動內存管理機制相關概念

開發 后端
.NET Framework自動內存管理在實際應用中是比較重要的一個步驟。掌握其中的運行機制對于我們的代碼編程是一個非常關鍵的操作。

我們在學習的過程中如何才能正確的掌握.NET Framework自動內存管理的機制呢?首先讓我們先從這篇文章介紹的內容開始學起。在.NET Framework中,內存中的資源(即所有二進制信息的集合)分為"托管資源"和"非托管資源".托管資源必須接受.NET Framework的CLR(通用語言運行時)的管理(諸如內存類型安全性檢查),而非托管資源則不必接受.NET Framework的CLR管理. (了解更多區別請參閱.NET Framework或C#的高級編程資料)#t#

托管資源在.NET Framework中又分別存放在兩種地方: "堆棧"和"托管堆"(以下簡稱"堆");規則是,所有的值類型(包括引用和對象實例)和引用類型的引用都存放在"堆棧"中,而所有引用所代表的對象實例都保存在堆中.

在C#中,釋放托管資源是可以自動通過"垃圾回收器"完成的(注意,"垃圾回收"機制是.NET Framework的特性,而不是C#的),但具體來說,.NET Framework自動內存管理中仍有些需要注意的地方:

1.值類型(包括引用和對象實例)和引用類型的引用其實是不需要什么"垃圾回收器"來釋放內存的,因為當它們出了作用域后會自動釋放所占內存(因為它們都保存在"堆棧"中,學過數據結構可知這是一種先進后出的結構);

2.只有引用類型的引用所指向的對象實例才保存在"堆"中,而堆因為是一個自由存儲空間,所以它并沒有像"堆棧"那樣有生存期("堆棧"的元素彈出后就代表生存期結束,也就代表釋放了內存),并且非常要注意的是,"垃圾回收器"只對這塊區域起作用;

3."垃圾回收器"也許并不像許多人想象的一樣會立即執行(當堆中的資源需要釋放時),而是在引用類型的引用被刪除和它在"堆"中的對象實例被刪除中間有個間隔,為什么呢? 因為"垃圾回收器"的調用是比較消耗系統資源的,因此不可能經常被調用!

(當然,用戶代碼可以用方法System.GC.Collect()來強制執行"垃圾回收器")

然而,.NET Framework自動內存管理中的大多數情況下,我們需要明確地在不執行"垃圾回收器"的情況下釋放托管資源(因為只需要釋放一部分但又是非常需要釋放的資源,但最好不要調用"垃圾回收器",因為"垃圾回收器"太浪費系統資源了),或需要釋放"非托管資源",這時候我們該怎么辦? 這是我們寫代碼的時候必須要考慮的問題("垃圾回收器"是系統自動實現的,一般情況不需要用戶干預),否則Windows系統會因為內存耗盡而...

現在,我來告訴怎么辦,那就是使用類的Dispose()方法釋放所有類型資源 和 使用析構方法釋放非托管資源!

1.Dispose()方法

要通過Dispose()方法來釋放資源,那么在類定義的時候執"System.IDisposable"接口,然后在類中必須包含這樣定義的方法"void Dispose()" (在Dispose()方法中就是用戶自己寫的釋放資源的代碼段),這樣一來,用戶就會知道可以通過人為地調用Dispose()方法來釋放資源. 不過需要注意的是,"垃圾回收器"并不是通過調用Dispose()方法來釋放托管資源的!

2.析構方法

在C#中定義析構方法的格式是" ~CLASS_NAME() ".非常需要注意的是,如果一個類中沒有使用到非托管資源,那么請一定不要定義析構方法,這是因為對象執行了析構方法,那么"垃圾回收器"在釋放托管資源之前要先調用析構方法,然后第二次才真正釋放托管資源,這樣一來,兩次刪除動作的花銷比一次大多的! (不過,即使你在類中已經定義了析構方法,仍然有辦法"屏蔽"它,這將在后面的代碼范例中說明) 在析構方法中,就是用戶自己寫的釋放非托管資源的代碼段.

下面使用一段代碼來示范Dispose()方法和析構方法如何使用:

 

  1. public class ResourceHolder : System.IDisposable  
  2. {  
  3. public void Dispose()  
  4. {  
  5. Dispose(true);  
  6. System.GC.SuppressFinalize(this);  
  7. // 上面一行代碼作用是防止"垃圾回收器"調用這個類中的方法  
  8. // " ~ResourceHolder() "  
  9. // 為什么要防止呢? 因為如果用戶記得調用Dispose()方法,那么  
  10. // "垃圾回收器"就沒有必要"多此一舉"地再去釋放一遍"非托管資源"了  
  11. // 如果用戶不記得調用呢,就讓"垃圾回收器"幫我們去"多此一舉"吧 ^_^  
  12. // 你看不懂我上面說的不要緊,下面我還有更詳細的解釋呢!  
  13. }  
  14. protected virtual void Dispose(bool disposing)  
  15. {  
  16. if (disposing)  
  17. {  
  18. // 這里是清理"托管資源"的用戶代碼段  
  19. }  
  20. // 這里是清理"非托管資源"的用戶代碼段  
  21. }  
  22. ~ResourceHolder()  
  23. {  
  24. Dispose(false);  
  25. }  

 

上面的.NET Framework自動內存管理代碼是一個典型的有兩種Dispose方法的類定義.
在.NET Framework中有很多系統類是用這種方法定義Dispose()方法的,例如:
MSDN中,System.Drawing.Brush.Dispose方法就是這樣定義的:

* 釋放由此 Brush 對象使用的所有資源。 *

* public void Dispose() *

* 該成員支持 .NET 框架結構,因此不適用于直接從代碼中使用。 *

* protected virtual void Dispose(bool); *

這里,我們必須要清楚,需要用戶調用的是方法Dispose()而不是方法Dispose(bool),然而,這里真正執行釋放工作的方法卻并不是Dispose(),而是Dispose(bool) ! 為什么呢?仔細看代碼,在Dispose()中,調用了Dispose(true),而參數為"true"時,作用是清理所有的托管資源和非托管資源;大家一定還記得我前面才說過,"使用析構方法是用來釋放非托管資源的",那么這里既然Dispose()可以完成釋放非托管資源的工作,還要析構方法干什么呢? 其實,析構方法的作用僅僅是一個"備份"!

嚴格地說,.NET Framework自動內存管理中凡執行了接口"IDisposable"的類,那么只要程序員在代碼中使用了這個類的對象實例,那么早晚得調用這個類的Dispose()方法,同時,如果類中含有對非托管資源的使用,那么也必須釋放非托管資源! 可惜,如果釋放非托管資源的代碼放在析構方法中(上面的例子對應的是 " ~ResourceHolder() "),那么程序員想調用這段釋放代碼是不可能做到的(因為析構方法不能被用戶調用,只能被系統,確切說是"垃圾回收器"調用),所以大家應該知道為什么上面例子中"清理非托管資源的用戶代碼段"是在Dispose(bool)中,而不是~ResourceHolder()中!

不過不幸的是,并不是所有的程序員都時刻小心地記得調用Dispose()方法,萬一程序員忘記調用此方法,托管資源當然沒問題,早晚會有"垃圾回收器"來回收(只不過會推遲一會兒),那么非托管資源呢?它可不受CLR的控制啊!難道它所占用的非托管資源就永遠不能釋放了嗎? 當然不是!我們還有"析構方法"呢! 如果忘記調用Dispose(),那么"垃圾回收器"也會調用"析構方法"來釋放非托管資源的!(多說一句廢話,如果程序員記得調用Dispose()的話,那么代碼"System.GC.SuppressFinalize(this);"則可以防止"垃圾回收器"調用析構方法,這樣就不必多釋放一次"非托管資源"了) 所以我們就不怕程序員忘記調用Dispose()方法了.

所以我說了這么一大堆的.NET Framework自動內存管理理由,綜合起來只有兩點:

*1.程序員們啊,千萬不要忘記調用Dispose()方法! (如果有的話 ^_^)

*2.萬一忘記,不要著急...還有救!!! 因為還有"垃圾回收器"幫我們自動調用析構方法!

責任編輯:曹凱 來源: 博客園
相關推薦

2010-09-27 13:26:31

JVM內存管理機制

2009-09-02 09:23:26

.NET內存管理機制

2010-01-06 18:21:24

2013-09-29 15:11:46

Linux運維內存管理

2010-07-23 09:34:48

Python

2022-06-01 16:01:58

MySQL內存管理系統

2010-09-26 13:23:13

JVM內存管理機制

2010-01-05 10:29:43

.NET Framew

2010-01-05 16:15:05

.NET Framew

2010-01-06 19:22:43

.NET Framew

2009-08-26 14:52:19

.NET Framew

2010-12-10 15:40:58

JVM內存管理

2011-06-29 17:20:20

Qt 內存 QOBJECT

2020-11-08 14:32:01

JavaScript變量內存管理

2022-02-28 10:25:17

Python參數傳遞拷貝

2016-10-09 14:41:40

Swift開發ARC

2010-01-05 13:45:15

.NET Framew

2010-01-05 10:17:35

.NET Framew

2020-08-18 19:15:44

Redis內存管理

2019-01-23 17:08:52

Python內存管理RealPython
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91天堂网 | 狠狠操狠狠操 | 天天想天天干 | 久久性av| 日韩精彩视频 | 涩涩片影院 | 亚洲国产激情 | a国产一区二区免费入口 | 成人免费大片黄在线播放 | 91国产精品| 中文在线一区 | 91pao对白在线播放 | 日韩欧美一区二区三区四区 | 久久综合一区 | 久久久久国产 | 欧美女优在线观看 | 欧洲亚洲一区二区三区 | 精品欧美一区二区三区久久久 | 69性欧美高清影院 | 欧美一区二区三区小说 | 日韩一区二区在线视频 | 久久99精品国产99久久6男男 | 一级毛片在线播放 | 亚洲人人| 成人激情视频免费在线观看 | 一区二区三区影院 | 日韩www| 日韩欧美第一页 | 碰碰视频 | 国产区在线免费观看 | 玖草资源 | 欧美精品成人影院 | 成年视频在线观看 | 国产视频一区在线观看 | 日韩精品一区二区三区中文在线 | 麻豆av一区二区三区久久 | 欧美激情精品久久久久 | 精品网站999| 在线国产中文字幕 | 美女视频网站久久 | 亚洲欧美国产毛片在线 |