.NET 垃圾回收淺解

来源:https://www.cnblogs.com/supersnowyao/archive/2017/12/31/8159244.html
-Advertisement-
Play Games

在說明垃圾回收的實現機制之前,先說明一下垃圾回收存在的背景。 垃圾回收器(GC)是.NET平臺中一個很重要的組成部分,.NET垃圾回收機制降低了編寫程式的複雜程度,使程式員不用耗費精力去處理析構,成功的將記憶體管理從程式的編寫時,脫離到運行時。 一、析構函數 析構函數的作用主要是釋放類在構造函數中以及 ...


      在說明垃圾回收的實現機制之前,先說明一下垃圾回收存在的背景。

      垃圾回收器(GC)是.NET平臺中一個很重要的組成部分,.NET垃圾回收機制降低了編寫程式的複雜程度,使程式員不用耗費精力去處理析構,成功的將記憶體管理從程式的編寫時,脫離到運行時。

一、析構函數

     析構函數的作用主要是釋放類在構造函數中以及類生命周期中所獲取的資源,比如需要釋放互斥鎖、由操作符NEW所分配的對象記憶體等。當然析構函數也不只限於釋放資源,一般析構函數可以執行在最後一次操作對象後所需要執行的任何操作。

二、關於垃圾回收

     在.NET FRAMEWORK框架中,記憶體中的資源(二進位信息)分為“托管資源”和“非托管資源”,其中托管資源必須受到CLR的管理(如強類型安全檢查),非托管資源需要手動釋放。

     托管資源分別存放在兩個地方,一個是“堆棧”,另外一個是“托管堆”。值類型和引用類型的引用存放在“堆棧”中,而引用類型所代表的對象存在於“托管堆”中。

三、垃圾回收演算法

     CLR中的每個對象有兩個開銷欄位:類型對象指針,同步塊索引。

     (1)、對於對象生存期的管理,有的系統採用“引用計數演算法”,而對於微軟的COM用的就是此演算法。具體演算法就是每個對象都在維護著一個記憶體欄位,用來統計有多少個位置在使用該對象,當某一個位置不在使用該對象後,該記憶體欄位中的計數就減一,直至該欄位最後為0,當值為0時,該對象就可以從記憶體中刪除掉了。

但是該演算法最大的問題就是處理不好迴圈引用。

     (2)、CLR使用的演算法為“引用跟蹤演算法”,此演算法只關心引用類型的變數,因為只有這種變數才能引用堆上的對象。可以將引用類型的變數稱為根。

     此演算法是當CLR進行GC時,首先暫停所有線程,以防止線程CLR在檢查期間訪問對象並更改其狀態。

     A、CLR遍歷堆中的所有對象,並將同步塊索引欄位中的一位設為0,這表明所有對象可以被刪除。

     B、查看所有活動根,並查看它們引用了哪些對象。如果一個根包括NULL,則忽略此根繼續檢查下一個根。

     C、如果某根引用了堆上的對象,則會標記此對象,在該對象的同步塊索引中的位設為1。此對象被標記後,CLR會檢查這個對象中的根,標記它們引用的對象。若發現對象已經標記,則不再重新檢查該對象,也就避免了因為迴圈引用而產生死迴圈。

     D、檢查完畢後,堆中的對象要麼被標記,要麼未被標記,然後CLR進入GC的“壓縮”階段。(需要註意的是,靜態欄位引用的對象會一直存在,直到用於載入類型的APPDOMAIN卸載為止。內在泄漏的一個重要原因就是靜態欄位引用某個對象集合,然後不停的向對象集合添加數據項。所以,要儘量避免使用靜態欄位。)

四、代的機制

     CLR的GC是基於代的垃圾回收器,對於垃圾回收做了以下假設:

     1、對象越新,生存期越短。

     2、對象越老,生存期越長。

     3、回收堆的一部分,速度快於回收整個堆。

五、垃圾回收觸發條件

     1、當CLR檢測到第0代超過記憶體預算時,則觸發一次GC。

     2、代碼顯式調用GC的靜態方法Collect。大多時候要避免使用此方法。從第2代開始回收。

     3、當使用CreateMemoryResourceNotification或者QueryMemoryResourceNotification來監視系統的總體記憶體使用情況時,如果Windows報告記憶體低,則CLR會強制垃圾回收以釋放死對象,減小進程工作集。

     4、CLR卸載APPDOMAIN時,CLR會認為一切都不是根。

     5、CLR正在關閉,Windows將會回收進程中的全部記憶體。

六、垃圾回收模式

     有兩種基本GC模式:工作站、伺服器。應用程式預設使用以“工作站”GC模式運行。若想使用伺服器回收器,可以在配置文件中添加gcServer元素(gcServer=true)。

     另外還有兩種子模式:併發(預設)和非併發。一般併發垃圾回收器消耗的記憶體通常比使用非併發垃圾回收器多。若要使用非併發回收器,可在配置文件中創建gcConcurrent元素(enable=false)

七、需要特殊清理的類型

     針對一些本機資源的終結,CLR除了回收對象記憶體之前,還需要在Finalize中釋放本機資源,如下所示:

1  public class sometype()
2  {
3     ~sometype()//此為一個finalize方法
4  }  

     需要註意的是,finalize方法的執行時間是控制不了的。同時,CLR不保證多個finalize方法的調用順序,在finalize方法中最好不要訪問定義了finalize方法的其他類型的對象。調用靜態方法也要小心。由於此方法是為釋放本機資源而設計的,所以要謹慎使用。

如果想創建封裝了本機資源的類型時,需要繼承Safehandle。

     另外一種釋放本機資源的方法是Dispose,調用此方法只是控制這個清理動作的發生時間。同時,調用此方法並不會將托管對象從托管堆中刪除。若決定顯示調用Dispose方法,建議將調用放在一個異常處理finally塊中。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 為什麼需要泛型? 試想你需要一個簡單的容器類,或者說句柄類,比如要存放一個蘋果的籃子,那你可以這樣簡單的實現: 這樣一個簡單的籃子就實現了,但問題是它只能存放蘋果,之後又出現了另外的一大堆水果類,那你就不得不為這些水果類分別實現容器: 然後你發現你其實在做大量的重覆勞動。所以你幻想你的語言編譯器要是 ...
  • 1 兩種文本編輯器:Sublime Text 、 Notepad++ 執行python文件,在命令行中,切換到該Python文件所在的目錄下,然後輸入 Python ###.py命令就可以了。 2 Python的交互模式和直接運行.py文件有什麼區別呢? 直接輸入python進入交互模式,相當於啟動 ...
  • 本文主要介紹Spring中, 1 Spring JDBC 2 使用註解方式管理事務的傳播行為 3 採用XML 方式配置事務 4 SH 整合 5 SSH 整合 一、Spring JDBC 1) 導包 , 使用myeclipse2014, 添加與持久化相關的包 2) 引入名稱空間等 3) 配置數據源 4 ...
  • 描述 本片文章內容屬於ASP.NET MVC系列視圖篇,主要講解View,大致內容如下: 1.Views文件夾講解 2.View種類 3.Razor語法 4.對視圖的基本操作 一 Views文件夾 (一) Views文件夾下常用文件種類 分析: 1.ASP.NET MVC頁面基本被放在Views文件 ...
  • 前言 學習這件事情是一個習慣,不能停。。。另外這篇已經看過兩個月過去,但覺得有些事情不總結跟沒做沒啥區別,遂記下此文 1.CLR線程池基礎 2.ThreadPool的簡單使用練習 3.執行上下文 4.協作式取消和超時,System.Threading.CancellationTokenSource的 ...
  • 一、引言 開始寫這篇博客前,已經嘗試練習過好多次Docker環境安裝,.Net Core環境安裝了,在這裡替騰訊雲做一個推廣,假如我們想學習、練手.net core 或是Docker卻苦於沒有開發環境,伺服器也不想買,那麼我們可以使用騰訊雲提供的開發者實驗,每個實驗項目每天可以免費使用一次,時常45 ...
  • 2017即將過去,這幾天總想寫點什麼,記錄下這多事的一年。 17年對於我來說是意義非凡的一年,是歷經磨難的一年。這一年發生了很多事,是在20多年成長學習生涯中收穫最多的一年,也是做決定最多的一年。我的社會角色從一個學生變成了一個職場人仕。生活狀態從每天上學,放學,做作業過度到上班,下班,睡覺。對於這... ...
  • 1、Global.asax 文件包含響應 ASP.NET 或HTTP模塊所引發的應用程式級別和會話級別事件的代碼。 2、Global.asax.cs中的方法的含義: Application_Init:在每一個HttpApplication實例初始化的時候執行 Application_Disposed ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...