PerfView專題 (第三篇):如何尋找 C# 中的 VirtualAlloc 記憶體泄漏

来源:https://www.cnblogs.com/huangxincheng/archive/2022/08/14/16584911.html
-Advertisement-
Play Games

一:背景 上一篇我們聊到瞭如何用 PerfView 去偵察 NTHeap 的記憶體泄漏,這種記憶體泄漏往往是用 C 的 malloc 或者 C++ 的 new 分配而不釋放所造成的,這一篇我們來聊一下由 VirtualAlloc 方法造成的泄漏如何去甄別? 瞭解 VirtualAlloc 的朋友肯定說, ...


一:背景

上一篇我們聊到瞭如何用 PerfView 去偵察 NTHeap 的記憶體泄漏,這種記憶體泄漏往往是用 C 的 malloc 或者 C++ 的 new 分配而不釋放所造成的,這一篇我們來聊一下由 VirtualAlloc 方法造成的泄漏如何去甄別?

瞭解 VirtualAlloc 的朋友肯定說, C# 這種高層語言怎麼可能會用 VirtualAlloc 呢?即便是 C++ 大概率也不會用這個,其實這麼說還是世面見少了,經歷的案例太少,接下來我們就來簡要聊一聊。

二: C# 中真的會用 VirtualAlloc 嗎

常規的 C# 記憶體分配確實不會直接調用 VirtualAlloc,但那些圖形圖形的工具方法肯定會直接用的,比如說 Bitmap,如果不信的話,我可以讓你眼見為實,先上一段代碼。


        static void Main(string[] args)
        {
            for (int i = 0; i < int.MaxValue; i++)
            {
                Test2();
                Console.WriteLine(i);
            }

            Console.ReadLine();
        }

        public static void Test2()
        {
            int width = 1000;
            int height = 1000;

            Bitmap bitmap = new Bitmap(width, height);

            string path = @"D:\test\1.jpg";

            bitmap.Save(path);
        }

這段代碼中我會生成 1000x1000 的圖片,接下來用 bp KernelBase!VirtualAlloc 去做一個攔截。


0:009> bp KernelBase!VirtualAlloc
0:009> g
Breakpoint 0 hit
KERNELBASE!VirtualAlloc:
00007ffd`0d53f9e0 4883ec38        sub     rsp,38h
0:000> k
 # Child-SP          RetAddr               Call Site
00 00000000`001ce828 00007ffc`eaaf4483     KERNELBASE!VirtualAlloc
01 00000000`001ce830 00007ffc`eaaf35fb     gdiplus!GpMemoryBitmap::AllocBitmapData+0x137
02 00000000`001ce870 00007ffc`eaacded1     gdiplus!GpMemoryBitmap::AllocBitmapMemory+0x3f
03 00000000`001ce8b0 00007ffc`eaacddf2     gdiplus!GpMemoryBitmap::InitNewBitmap+0x49
04 00000000`001ce8f0 00007ffc`eaacdf6f     gdiplus!CopyOnWriteBitmap::CopyOnWriteBitmap+0x8a
05 00000000`001ce930 00007ffc`eaace074     gdiplus!GpBitmap::GpBitmap+0x6b
06 00000000`001ce970 00007ffc`357d8143     gdiplus!GdipCreateBitmapFromScan0+0xc4
07 00000000`001ce9d0 00007ffc`357e8eb1     0x00007ffc`357d8143
08 00000000`001ceaa0 00007ffc`357d3288     System_Drawing_Common!System.Drawing.Bitmap..ctor+0x31 [_/src/libraries/System.Drawing.Common/src/System/Drawing/Bitmap.cs @ 80] 
09 00000000`001ceaf0 00007ffc`357d2974     ConsoleApp7!ConsoleApp7.Program.Test2+0x58 [D:\net6\ConsoleApp1\ConsoleApp7\Program.cs @ 27] 
...

從輸出中可以看到在 Program.Test2調用的過程中果然被 VirtualAlloc 攔住了,而區區 200 多個 Bitmap 就已經分配了 1G 個記憶體,截圖如下:

而此時的 GCHeap 上才區區 1.7M


0:000> !eeheap -gc
Number of GC Heaps: 1
generation 0 starts at 0x0000000002941030
generation 1 starts at 0x0000000002941018
generation 2 starts at 0x0000000002941000
ephemeral segment allocation context: none
         segment             begin         allocated         committed    allocated size    committed size
0000000002940000  0000000002941000  0000000002ADFFE8  0000000002AE2000  0x19efe8(1699816)  0x1a1000(1708032)
Large object heap starts at 0x0000000012941000
         segment             begin         allocated         committed    allocated size    committed size
0000000012940000  0000000012941000  0000000012941018  0000000012942000  0x18(24)  0x1000(4096)
Pinned object heap starts at 0x000000001A941000
000000001A940000  000000001A941000  000000001A949C10  000000001A952000  0x8c10(35856)  0x11000(69632)
Total Allocated Size:              Size: 0x1a7c10 (1735696) bytes.
Total Committed Size:              Size: 0x1a2000 (1712128) bytes.
------------------------------
GC Allocated Heap Size:    Size: 0x1a7c10 (1735696) bytes.
GC Committed Heap Size:    Size: 0x1a2000 (1712128) bytes.

非常明顯的非托管泄漏。

三:如何用 Perfview 檢測

perfview 是一個非常好的運行時檢測工具,它也是根據 鉤子函數 攔截後看分配量來做一個權重,最終根據權重占比尋找到問題函數調用棧。

勾選上 VirtualAlloc 之後就可以點擊 Start Collection,5s 之後就會生成一個 統計報表 ,我們點擊 Memory -> Net Virtual Alloc Stacks 選項。

在彈出面板中選擇我們的程式,點擊 CallTree 面板,清除 GroupPats 分組,截圖如下:

從面板中可以看到,在記憶體分配權重總量上,Test2() 占比 97.9%, 說明確實是一個問題,而且是初始化 Bitmap 出來的。

到這裡, VirtualAlloc 的泄漏問題就找出來了, 如果用 WinDbg 分析的話,還需要開啟 ust 選項,也是記錄線程棧,但使用起來相對繁瑣。


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

-Advertisement-
Play Games
更多相關文章
  • 你必須非常努力,才能看起來毫不費力! 微信搜索公眾號[ 漫漫Coding路 ],一起From Zero To Hero ! 前言 日常 Go 開發中,Context 包是用的最多的一個了,幾乎所有函數的第一個參數都是 ctx,那麼我們為什麼要傳遞 Context 呢,Context 又有哪些用法,底 ...
  • 最近在看C語言代碼時碰到了這個問題,結合查找的資料對這C的知識點做了一下小結。寫了一份測試它們的代碼。test1函數穿了一個char* const的指針,如果對它增加,會報錯,它是只讀的。但是可以對指針所指位置的內容進行更改。test2函數測試的是const char類型的參數,test3函數測試的 ...
  • java中只有值傳遞 為什麼這麼說?兩個例子: public class Student { int sage = 20; String sname = "雲胡不歸"; public static void change(Student s1) { s1.sname = "荀一"; s1.sage ...
  • 一.元素操作 send_keys:賦值 clear:清空 click:點擊 from selenium import webdriver from selenium.webdriver.common.by import By driver = webdriver.Chrome() driver.ge ...
  • 前言 紅黑樹是一種特殊的B樹是B樹種2-3-4樹的一種特殊實現,紅黑樹保證了每個節點只會有兩個子節點,通過對每個節點進行染色,然後通過不同顏色的節點組合來分別代表2-3-4的2節點、3節點、4節點樹的情況。在學習紅黑樹之前,我們需要先去瞭解2-3-4樹。 一、 B樹 那麼如果想要對紅黑樹有一個較為深 ...
  • 書籍下載地址: Ansi Common Lisp 中文版|百度網盤 實用Common.Lisp編程 .pdf|百度網盤 LISP指令速查網站推薦: Simplified Common Lisp reference (jtra.cz) 我製作的表格: 由於Common LISP命令很多,經常忘,所以做 ...
  • #使用Nginx在 Linux 上托管 ASP.NET Core 6.0應用:GitHub Actions自動部署 前言 本文主要參考微軟這篇文檔而來 Host ASP.NET Core on Linux with Nginx,並使用Github Actions做CI&CD,部署到阿裡雲伺服器,所有 ...
  • 一、在使用WinDbg調試之前,我們先使用VS的調試功能 1、文件 -> 打開 -> 文件 -> 打開Dump文件 2、調試 -> 選項 -> 調試 -> 符號 -> 添加該應用的.pdb所在的文件夾 3、使用 僅限托管 進行調試 正常情況下會執行到拋出異常的位置,並且顯示異常詳情。 二、WinDb ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...