LOGO OA教程 ERP教程 模切知識交流 PMS教程 CRM教程 開發文檔 其他文檔  
 
網站管理員

一次 .NET 性能優化之旅:將 GC 壓力降低 99%

freeflydom
2025年7月2日 9:34 本文熱度 501

前言:問題的浮現

最近,我使用 ScottPlot 庫開發一個頻譜分析應用。應用的核心功能之一是實時顯示頻譜圖,這可以看作是一個高頻刷新熱力圖(Heatmap)。然而,在程序運行一段時間后,我注意到整體性能開始逐漸下降,界面也出現了卡頓。直覺告訴我,這背后一定隱藏著性能瓶頸。

分析:探尋性能瓶頸

面對性能問題,我首先打開了 Visual Studio 的診斷工具,重點關注計數器(Counters)的變化。

 

VS 診斷工具

上圖揭示了幾個嚴重的問題:

  1. 1. GC 頻繁:進程內存圖表中,GC(垃圾回收)標記幾乎連成一片,表明垃圾回收異常頻繁。
  2. 2. GC 耗時過長:% Time in GC since last GC 的值非常高,說明 GC 占用了大量的 CPU 時間。
  3. 3. 高內存分配率:Allocation Rate 居高不下,意味著程序在以極高的速率分配內存。

顯然,問題出在 GC 上。但究竟是哪部分代碼導致了如此巨大的 GC 壓力呢?

定位:追蹤 GC 的“元兇”

為了找出問題的根源,我使用了 Visual Studio 的性能探查器(Performance Profiler),并選擇了 .NET 對象分配跟蹤(.NET Object Allocation Tracking)模式。

在程序運行一段時間后,我停止了分析,并查看了分配(Allocations)選項卡。結果令人震驚:System.Double 類型的分配次數和字節數都異常巨大。這正是導致 GC 頻繁的“元兇”。

通過調用堆棧,我迅速定位到了問題代碼:

 

調用堆棧

函數名                                          分配        字節          模塊名稱
+ ScottPlot.NumericConversion.Clamp<T>(T, T, T)    3,592,245    86,213,880    scottplot

所有的矛頭都指向了 ScottPlot.NumericConversion.Clamp<T>(T, T, T) 這個函數。

探究:泛型與裝箱的“陷阱”

為了弄清真相,我翻閱了 ScottPlot 的源代碼,并梳理了整個調用流程:

  1. 1. 在繪制熱力圖時,程序會調用 NumericConversion.Clamp 函數,將數據歸一化到 0-1 的范圍內。
  2. 2. 接著,程序會根據歸一化后的值,從顏色映射表(ColorMap)中獲取對應的顏色。
public Color GetColor(double position)
{
    position = NumericConversion.Clamp(position, 01);
    int index = (int)((Colors.Length - 1) * position);
    return Colors[index];
}

問題就出在 NumericConversion.Clamp 函數的實現上:

public static T Clamp<T>(T input, T min, T max) where T : IComparable
{
    if (input.CompareTo(min) < 0) return min;
    if (input.CompareTo(max) > 0) return max;
    return input;
}

這是一個泛型方法,并且 double 是值類型。當 double 作為參數傳遞給這個泛型方法時,會發生裝箱(boxing),即 double 被轉換為 IComparable 接口。在每秒數萬次的調用下,這會導致頻繁的堆分配,從而引發巨大的 GC 壓力。

深究:為何會發生裝箱?

首先感謝兩位大神的指出,問題的根源在于 Clamp<T> 方法的泛型約束 where T : IComparable,修改為使用 where T : IComparable<T>就可以避免裝箱的問題。但為什么這個約束會導致裝箱呢?

答案隱藏在 IComparable 接口的定義之中。讓我們來看一下它的 CompareTo 方法:

// 非泛型版本
public interface IComparable
{
    int CompareTo(object? obj);
}

正如你所見,CompareTo 方法接受一個 object 類型的參數。當我們將像 double 這樣的值類型傳遞給它時,CLR 為了匹配方法簽名,必須將其轉換為引用類型。這個從值類型到 object 的轉換過程,就是裝箱。每一次裝箱都會在托管堆上分配一小塊內存,在高頻調用的場景下,這會迅速累積成巨大的內存壓力,迫使 GC 頻繁介入。

.NET 同時為我們提供了泛型版本的 IComparable<T> 接口:

// 泛型版本
public interface IComparable<in T>
{
    int CompareTo(T? other);
}

看到區別了嗎?這個版本的 CompareTo 方法接受的是一個類型為 T 的參數。由于 double 等基礎值類型已經實現了 IComparable<double>,編譯器可以進行類型匹配,從而直接調用,完全避免了裝箱操作。

因此,如果 ScottPlot 的源代碼將約束改為 where T : IComparable<T>,就可以從根本上解決裝箱導致的這個性能問題。不過,直接使用對應值類型的重載的性能還是會大幅的高于IComparable的版本,具體原因這里就不展開講了。

優化:小改動,大提升

找到了問題的根源,解決方案也就水到渠成了。我為 Clamp 函數添加了一個 double 類型的重載版本,從而避免了裝箱操作:

public static double Clamp(double input, double min, double max)
{
    if (input < minreturn min;
    if (input > maxreturn max;
    return input;
}

測試:驗證優化效果

為了驗證優化效果,我使用 LinqPad 和 BenchmarkDotNet 進行了性能測試。

#load "BenchmarkDotNet"
void Main()
{
    RunBenchmark();
}
privatedoublevalue = 0.75;
privatedouble min = 0.0;
privatedouble max = 1.0;
[Benchmark]
public double Clamp_Double()
    => NumericConversion.Clamp(value, min, max);
[Benchmark]
public double Clamp_Generic()
    => NumericConversion.Clamp<double>(value, min, max);
publicstaticclassNumericConversion
{
    public static double Clamp(double valuedouble min, double max)
        => value < min ? min : (value > max ? max : value);
    public static T Clamp<T>(T input, T min, T maxwhere T : IComparable
    {
        if (input.CompareTo(min) < 0return min;
        if (input.CompareTo(max) > 0return max;
        return input;
    }
}

測試結果如下:

 

性能測試結果

從上圖可以看出,新添加的 Clamp_Double 方法在性能上遠超泛型版本。

再次打開 Visual Studio 的診斷工具,GC 壓力幾乎消失了:

 

優化后診斷工具

總結:性能優化的啟示

通過對 GC 壓力的分析和優化,我成功解決了程序中的性能瓶頸。這次優化的核心在于,通過為 NumericConversion.Clamp 函數添加 double 類型的重載,避免了高頻調用下的裝箱操作,從而顯著提升了性能,并將 GC 壓力降低了 99% 以上。

這次經歷不僅提升了程序的運行效率,也為我未來的性能調優工作積累了寶貴的經驗。

目前,我已經將針對 ScottPlot 源碼的修改提交了 PR:https://github.com/ScottPlot/ScottPlot/pull/4985

?轉自https://www.cnblogs.com/Cookies-Tang/p/18956241


該文章在 2025/7/2 9:34:40 編輯過
關鍵字查詢
相關文章
正在查詢...
點晴ERP是一款針對中小制造業的專業生產管理軟件系統,系統成熟度和易用性得到了國內大量中小企業的青睞。
點晴PMS碼頭管理系統主要針對港口碼頭集裝箱與散貨日常運作、調度、堆場、車隊、財務費用、相關報表等業務管理,結合碼頭的業務特點,圍繞調度、堆場作業而開發的。集技術的先進性、管理的有效性于一體,是物流碼頭及其他港口類企業的高效ERP管理信息系統。
點晴WMS倉儲管理系統提供了貨物產品管理,銷售管理,采購管理,倉儲管理,倉庫管理,保質期管理,貨位管理,庫位管理,生產管理,WMS管理系統,標簽打印,條形碼,二維碼管理,批號管理軟件。
點晴免費OA是一款軟件和通用服務都免費,不限功能、不限時間、不限用戶的免費OA協同辦公管理系統。
Copyright 2010-2025 ClickSun All Rights Reserved

黄频国产免费高清视频,久久不卡精品中文字幕一区,激情五月天AV电影在线观看,欧美国产韩国日本一区二区
欧美日韩中文字幕久久久不卡 | 亚洲综合另类在线A∨ | 亚洲国产日韩a在线亚洲 | 亚洲精品国产高清一线久久 | 五月婷婷免费视频 | 日韩综合网在线视频免费 |