DotNet開發之反射技術詳解—動態庫的加載
反射是一種強大的工具,它允許程序在運行時獲取關于類型、成員和程序集的信息,并動態地創建、調用和修改代碼。今天,我們將深入探討反射的其他用法,以幫助您更好地利用這一技術。
一、Assembly.Load(),Assembly.LoadFrom(),Assembly.LoadFile()的區別
1.Assembly.Load
- Assembly.Load(string assemblyString):通過程序集的完全限定名稱(包括版本號、公鑰令牌等)或者簡單名稱來加載程序集。例如:Assembly.Load("MyAssembly, Versinotallow=1.0.0.0, Culture=neutral, PublicKeyToken=null")或Assembly.Load("MyAssembly")。
- Assembly.Load(byte[] rawAssembly):從字節數組中加載程序集。這對于從網絡或其他非傳統來源加載程序集很有用。
示例用法:
// 根據程序集名稱加載程序集
Assembly assembly = Assembly.Load("MyAssembly");
// 從字節數組加載程序集
byte[] assemblyBytes = File.ReadAllBytes("MyAssembly.dll");
Assembly assembly = Assembly.Load(assemblyBytes);
2.Assembly.LoadFrom
- Assembly.LoadFrom(string assemblyFile):從指定路徑加載程序集文件。
- Assembly.LoadFrom(string assemblyFile, Evidence securityEvidence):基于安全證據加載程序集文件。
示例用法:
// 從文件路徑加載程序集
Assembly assembly = Assembly.LoadFrom("C:\\MyAssemblies\\MyAssembly.dll");
// 基于安全證據加載程序集
Evidence evidence = new Evidence();
Assembly assembly = Assembly.LoadFrom("C:\\MyAssemblies\\MyAssembly.dll", evidence);
3.Assembly.LoadFile
- Assembly.LoadFile(string path):從指定路徑加載程序集文件。與LoadFrom不同,它會創建一個新的加載上下文,程序集將在該上下文中加載并解析。
示例用法:
// 從文件路徑加載程序集
Assembly assembly = Assembly.LoadFile("C:\\MyAssemblies\\MyAssembly.dll");
二、加載程序集的異常情況
需要注意的是,在使用這些方法加載程序集時,你需要處理可能出現的異常情況。以下可能會拋出以下幾種異常:
1.FileNotFoundException:
- 當指定的程序集文件無法找到時,會拋出此異常。
- 可能的原因包括:文件路徑錯誤、文件不存在或不可訪問等。
2.FileLoadException:
- 當無法加載指定的程序集文件時,會拋出此異常。
- 可能的原因包括:無效的程序集文件、程序集版本不兼容、程序集依賴項無法解析等。
3.BadImageFormatException:
- 當嘗試加載無效的程序集文件格式時,會拋出此異常。
- 可能的原因包括:程序集文件不是有效的CLR程序集、程序集文件被損壞或篡改等。
4.SecurityException:
- 當沒有足夠權限來加載程序集時,會拋出此異常。
- 可能的原因包括:缺少適當的安全權限、程序集未經數字簽名等。
5.ReflectionTypeLoadException:
- 當加載程序集時出現類型加載異常時,會拋出此異常。
- 可能的原因包括:程序集中某個類型的依賴項無法解析、類型初始化失敗等。
6.UnauthorizedAccessException:
- 當沒有足夠的權限訪問程序集文件時,會拋出此異常。
- 可能的原因包括:訪問權限限制、文件被占用或鎖定等。
這些異常通常會在使用Assembly.Load()方法加載程序集時的各種錯誤情況下拋出。為了確保程序的穩定性和可靠性,建議在使用Assembly.Load()方法時使用適當的異常處理機制來捕獲和處理這些異常。此外,當使用這些方法加載程序集時,還要注意程序集的版本和依賴關系,確保加載的程序集符合預期,并且不會引發版本沖突或依賴項缺失的問題。
三、DLL加載規則
我相信很多使用反射技術或者不使用反射技術,加載程序集的時候,都曾經遇到過BadImageFormatException的問題。在加載DLL(動態鏈接庫)時,系統會按照一定的路徑規則進行搜索。以下是通常情況下的DLL加載路徑規則:
應用程序目錄:系統首先會搜索執行當前代碼的應用程序目錄。這是最常見的搜索位置,通常會將DLL文件放置在應用程序的根目錄或相關子目錄中。
系統目錄:如果在應用程序目錄中未找到DLL文件,則系統會搜索系統目錄。系統目錄通常是指 %SystemRoot%\System32(在大多數情況下為 C:\Windows\System32)。請注意,32位應用程序在64位操作系統上可能會搜索 %SystemRoot%\SysWOW64 目錄。
Windows目錄:如果在系統目錄中未找到DLL文件,則系統會搜索Windows目錄。Windows目錄通常是指 %SystemRoot%\(在大多數情況下為 C:\Windows)。
當前工作目錄:如果在以上路徑中未找到DLL文件,則系統會搜索當前工作目錄。當前工作目錄是指應用程序正在運行的目錄。
環境變量指定的路徑:系統還會搜索在系統環境變量 PATH 中指定的路徑。PATH 變量包含一系列目錄路徑,用于指示系統在其中搜索可執行文件和DLL文件。
請注意,搜索DLL的順序是按照上述規則從前往后進行的,一旦找到匹配的DLL文件,搜索過程就會停止。此外,還可以使用以下方法來影響DLL的加載路徑:
- 使用絕對路徑:指定DLL的完整路徑,確保系統可以直接訪問該路徑下的DLL文件。
- 使用相對路徑:指定相對于應用程序或當前工作目錄的相對路徑。在這種情況下,需要確保指定的相對路徑是正確的。