大家好,我是沙漠盡頭的狼。
本方首发于Dotnet9,介绍使用dnSpy调试第三方.NET库源码,行文目录:
- 安裝dnspy
- 編寫示例程式
- 調試示例程式
- 調試.net庫原生方法
- 總結
1. 安裝dnspy
dnspy是一款功能強大的.net程式反編譯工具,可以對.net程式進行反編譯,代替庫文檔的功能,代碼丟失或者損壞可直接恢復,所以能在完全沒有源碼的情況下即時調試程式,甚至還能修改程式!
github有二進位可執行程式下載地址,也有源碼可以自行編譯,本文使用前者,github地址是:https://github.com/dnSpy/dnSpy

2. 編寫示例程式
示例為一個桌面程式,輸入數字,右側回顯輸入的數字是奇數還是偶數:

示例代碼比較簡單,界面綁定和viewmodel關係截圖看完所有:

奇偶判断由类TestTool的TellMeOddEven方法返回,再回看回显,咦,0是奇数?1是偶数?
TestTool类是其他库定义,我假装你没有源码哈,虽然你有:

類具體定義如下:
namespace TestDll;
public class TestTool
{
public string TellMeOddEven(int number)
{
if (number % 2 == 1)
{
return $"{number}是偶数";
}
return $"{number}是奇数";
}
}
3. 調試示例程式
打开dnSpy,将主程序引用的TestDll拖入:

可以看到反編譯後的代碼:

反編譯出來的方法定義會和第三方源碼可能不同,以下是一些可能導致不同反編譯結果的因素:
編譯器優化:不同版本的編譯器可能會對代碼進行不同的優化,例如使用不同的算法、數據結構或者代碼重排等。這些優化可能會導致反編譯出來的代碼結構和順序不同,本文示例使用 .net 8開發,.net framework編譯的庫可能反編譯出來與源碼幾乎一致。
反編譯工具更新:dnspy本身也會不斷更新,以適應新的.net版本和編譯器特性。這些更新可能會改變反編譯算法和策略,從而導致不同版本的dnspy反編譯結果不一致。
代碼簡單,對比源碼和反編譯的代碼查看,對整型入參除2取余,如果等於1判斷為偶數,否則為奇數,當然這是錯的,假如代碼邏輯複雜,可以用dnspy調試。
運行測試程式,並在dnspy中給方法打斷點,在調試菜單附加測試程式,就和vs中操作類似:

4. 調試.net庫方法
上面調試示例程式的方法可用於其他第三方.net庫,那麼.net自身庫方法呢?
方法類似,找到.net庫對應類、對應方法,運行目標程式,然後打斷點。. net庫方法這樣找:點擊【文件】》【從gac打開】=》搜索目標庫,雙擊庫,再查找目標方法,後面調試步驟就是一樣了:

5. 總結
- 技術交流加群請添加站長微信號:codewf
- 文中示例代码:MultiVersionLibrary
dnSpy很强大的,还能直接监视第三方代码的变量、修改值等,就和你使用VS开发自己的程序一样,了解更多用法还请查看文章开头给的链接https://github.com/dnSpy/dnSpy, 这篇大佬的文章也不错,建议看看:《神器如 dnSpy,无需源码也能修改 .NET 程序》。
對了,示例程式中奇偶數判斷不對,我又沒代碼我想糾正怎麼辦?
解决这个问题,上面大佬的文章您可以拜读了,下一篇站长继续讲解第三方库拦截,能实现不修改第三方库达到修改方法逻辑和返回结果的效果,可以提前预习快学会这个技能-.NET API拦截技法,当然下一篇会有新知识点:非公有类非公有方法拦截技法。
以原倉庫兩張dnspy調試第三方庫的動圖結束本文:

