如何在沒有第三方.NET庫原始碼的情況下除錯第三方庫程式碼?

如何在沒有第三方.NET庫原始碼的情況下除錯第三方庫程式碼?

藉助dnSpy除錯第三方庫程式碼

最後更新 2023/9/21 下午10:00
沙漠尽头的狼
預計閱讀 4 分鐘
分類
.NET
標籤
.NET C#

大家好,我是沙漠盡頭的狼。

本文首發於Dotnet9,介紹使用dnSpy除錯第三方.NET程式庫原始碼,行文目錄:

  • 安裝dnSpy
  • 撰寫範例程式
  • 除錯範例程式
  • 除錯.NET程式庫原生方法
  • 總結

1. 安裝dnSpy

dnSpy是一款功能強大的.NET程式反編譯工具,可以對.NET程式進行反編譯,代替程式庫文件的功能,程式碼遺失或損壞可直接還原,所以能在完全沒有原始碼的情況下即時除錯程式,甚至還能修改程式!

GitHub有二進位可執行檔下載位址,也有原始碼可以自行編譯,本文使用前者,GitHub位址是:https://github.com/dnSpy/dnSpy

2. 撰寫範例程式

範例為一個桌面程式,輸入數字,右側回顯輸入的數字是奇數還是偶數:

範例程式碼比較簡單,介面繫結和ViewModel關係截圖看完所有:

奇偶判斷由類別TestToolTellMeOddEven方法回傳,再回頭看回顯,咦,0是奇數?1是偶數?

TestTool類別是其他程式庫定義,我假裝你沒有原始碼哈,雖然你有:

類別具體定義如下:

namespace TestDll;

public class TestTool
{
    public string TellMeOddEven(int number)
    {
        if (number % 2 == 1)
        {
            return $"{number}是偶數";
        }

        return $"{number}是奇數";
    }
}

3. 除錯範例程式

開啟dnSpy,將主程式引用的TestDll拖入:

可以看到反編譯後的程式碼:

反編譯出來的方法定義會和第三方原始碼可能不同,以下是一些可能導致不同反編譯結果的因素:

  1. 編譯器最佳化:不同版本的編譯器可能會對程式碼進行不同的最佳化,例如使用不同的演算法、資料結構或者程式碼重排等。這些最佳化可能會導致反編譯出來的程式碼結構和順序不同,本文範例使用 .NET 8 開發,.NET Framework 編譯的程式庫可能反編譯出來與原始碼幾乎一致。

  2. 反編譯工具更新:dnSpy本身也會不斷更新,以適應新的.NET版本和編譯器特性。這些更新可能會改變反編譯演算法和策略,從而導致不同版本的dnSpy反編譯結果不一致。

程式碼簡單,對比原始碼和反編譯的程式碼查看,對整型入參除2取餘數,如果等於1判斷為偶數,否則為奇數,當然這是錯的,假如程式碼邏輯複雜,可以用dnSpy除錯。

執行測試程式,並在dnSpy中給方法中斷點,在除錯功能表附加測試程式,就和VS中操作類似:

4. 除錯.NET程式庫方法

上面除錯範例程式的方法可用於其他第三方.NET程式庫,那麼.NET自身程式庫方法呢?

方法類似,找到.NET程式庫對應類別、對應方法,執行目標程式,然後中斷點。.NET程式庫方法這樣找:點選【檔案】》【從GAC開啟】=》搜尋目標程式庫,雙擊程式庫,再查詢目標方法,後面除錯步驟就一樣了:

5. 總結

dnSpy很強大的,還能直接監視第三方程式碼的變數、修改值等,就和你使用VS開發自己的程式一樣,了解更多用法還請查看文章開頭給的連結https://github.com/dnSpy/dnSpy, 這篇大佬的文章也不錯,建議看看:《神器如 dnSpy,無需原始碼也能修改 .NET 程式》

對了,範例程式中奇偶數判斷不對,我又沒程式碼我想糾正怎麼辦?

解決這個問題,上面大佬的文章您可以拜讀了,下一篇站長繼續講解第三方程式庫攔截,能實現不修改第三方程式庫達到修改方法邏輯和回傳結果的效果,可以提前預習快學會這個技能-.NET API攔截技法,當然下一篇會有新知:非公有類別非公有方法攔截技法。

以原倉庫兩張dnSpy除錯第三方程式庫的動圖結束本文:

繼續探索

延伸閱讀

更多文章
同分類 / 同標籤 2026/2/7

AOT使用經驗總結

從專案建立伊始,就應養成良好的習慣,即只要添加了新功能或使用了較新的語法,就及時進行 AOT 發布測試。

繼續閱讀