從GC的SuppressFinalize方法帶你深刻認識Finalize底層運行機制

来源:https://www.cnblogs.com/huangxincheng/archive/2020/04/30/12811291.html
-Advertisement-
Play Games

如果你經常看開源項目的源碼,你會發現很多Dispose方法中都有這麼一句代碼: ,看過一兩次可能無所謂,看多了就來了興趣,這篇就跟大家聊一聊。 一:背景 1. 在哪發現的 相信現在Mysql在.Net領域中鋪的面越來越廣了,C 對接MySql的MySql.Data類庫的代碼大家可以研究研究,幾乎所有 ...


如果你經常看開源項目的源碼,你會發現很多Dispose方法中都有這麼一句代碼: GC.SuppressFinalize(this); ,看過一兩次可能無所謂,看多了就來了興趣,這篇就跟大家聊一聊。

一:背景

1. 在哪發現的

相信現在Mysql在.Net領域中鋪的面越來越廣了,C#對接MySql的MySql.Data類庫的代碼大家可以研究研究,幾乎所有操作資料庫的幾大對象:MySqlConnection,MySqlCommand,MySqlDataReader以及內部的Driver都存在 GC.SuppressFinalize(this)代碼。


public sealed class MySqlConnection : DbConnection, ICloneable
{
    public new void Dispose()
    {
	    Dispose(disposing: true);
	    GC.SuppressFinalize(this);
    }
}

public sealed class MySqlCommand : DbCommand, IDisposable, ICloneable
{
    public new void Dispose()
    {
	    Dispose(disposing: true);
	    GC.SuppressFinalize(this);
    }
}

2. GC.SuppressFinalize 場景在哪裡

先看一下官方對這個方法的解釋,如下所示:

        //
        // Summary:
        //     Requests that the common language runtime not call the finalizer for the specified
        //     object.
        //
        // Parameters:
        //   obj:
        //     The object whose finalizer must not be executed.
        //
        // Exceptions:
        //   T:System.ArgumentNullException:
        //     obj is null.
        [ReliabilityContract(Consistency.WillNotCorruptState, Cer.Success)]
        [SecuritySafeCritical]
        public static void SuppressFinalize(object obj);

意思就是說: 請求 CLR 不要調用指定對象的終結器,如果你對終結器的前置基礎知識不足,那這句話肯定不是很明白,既然都執行了Dispose,說明非托管資源都被釋放了,怎麼還壓制CLR不要調用Finalize呢?刪掉和不刪掉這句代碼有沒有什麼嚴重的後果,GC類的方法誰也不敢動哈。。。 為了徹底講清楚,有必要說一下Finalize整個原理。

二:資源管理

我們都知道C#是一門托管語言,它的好處就是不需要程式員去關心記憶體的分配和釋放,由CLR統一管理,這樣編程門檻大大降低,天下攘攘皆為利來,速成系的程式員就越來越多~

1. 對托管資源和非托管資源理解

<1> 托管資源

這個很好理解,你在C#中使用的值類型,引用類型都是統一受CLR分配和GC清理。

<2> 非托管資源

在實際業務開發中,我們的代碼不可能不與外界資源打交道,比如說文件系統,外部網站,資料庫等等,就拿寫入文件的StreamWriter舉例,如下代碼:

        public static void Main(string[] args)
        {
            StreamWriter sw = new StreamWriter("xxx.txt");
            sw.WriteLine("....");
        }

為什麼能夠寫入文件? 那是因為我們的代碼是請求windows底層的Win32 Api幫忙寫入的,這就有意思了,因為這個場景有第三者介入,sw是引用類型受CLR管理,win32 api屬於外部資源和.Net一點關係都沒有,如果你在用完sw之後沒有調用close方法的話,當某個時候GC回收了托管堆上的sw後,這給被打開的win32 api文件句柄再也沒有人可以釋放了,資源就泄露了,如果沒看懂,我畫張圖:

三:頭疼的非托管資源解決方案

1. 使用析構函數

很多時候程式員就是在使用完類之後因為種種原因忘記了手動執行Close方法造成了資源泄露,那有沒有一種機制可以在GC回收堆對象的時候回調我的一個自定義方法呢?如果能實現就

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

-Advertisement-
Play Games
更多相關文章
  • 棧長在之前的文章分享過關於 Spring Boot 打包運行的兩篇文章: "運行 Spring Boot 應用的 3 種方式" "Spring Boot 怎麼打一個可執行 Jar 包?" 不過一直沒機會對 Spring Boot Maven 插件做詳細介紹。 Spring Boot 對 Maven ...
  • 0. 前言 這段時間完成了第二階段的Java作業練習,第一階段是入手,那麼這個階段則是這之後的學習打基礎;這幾次作業主要是加強我們對面向對象的封裝性、繼承性、多態性特征的理解,下麵是我對此次作業的總結分析。 1.作業過程總結 (1) 三次作業之間的知識迭代關係 第4次作業涉及數據檢驗及處理,類的繼承 ...
  • StringBuilder、StringBuffer源碼分析 StringBuilder源碼分析 類結構 StringBuilder使用final關鍵字修飾,和String一樣不可以被繼承 StringBuilder繼承AbstractStringBuilder並實現了Serializable和Ch ...
  • 首先看下ThinkPHP6官方手冊關於多應用的目錄結構: ├─app 應用目錄 │ ├─index 主應用 │ │ ├─controller 控制器目錄 │ │ ├─model 模型目錄 │ │ ├─view 視圖目錄 │ │ ├─config 配置目錄(優先) │ │ └─ ... 更多類庫目錄 ...
  • Gevent 是一個第三方庫,可以輕鬆通過gevent實現併發同步或非同步編程 ...
  • 程式目的:輸入年份和月份,查詢當月的日曆。弄著玩。程式界面:代碼如下:# coding:utf8from tkinter import *from calendar import *from time import *class APP: def __init__(self, master): fr... ...
  • 環境: 介紹 Quickuse.Caching 快速應用緩存組件,提供常用緩存使用方式,目前支持常用的 、`Redis Memcache` 運行時緩存 有時候也本稱作為伺服器緩存、進程緩存、站點緩存、程式緩存、本地緩存......各式各樣,我理解的其實他們都一個東西,都是在程式運行的時候才可以使用的 ...
  • 在之前的文章abp(net core)+easyui+efcore實現倉儲管理系統——入庫管理之九(四十五) 中我們已經實現了修改與刪除入庫單,今天來測試一下入庫單的修改與刪除功能。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...