背景
如果某個“功能”需要動態更新?這種動態更新,可能是需求驅動的,也可能是為了修改 BUG,面對這種場景,如何實現“熱插拔”呢?先解釋一下“熱插拔”:在系統運行過程動態替換某些功能,不用重啟系統進程。
幾種方案
- 腳本化:采用 Iron 或 集成其它腳本引擎。
- AppDomain:微軟的 Add In 框架就是為這個目的設計的。
- 分布式 + 負載平衡 :輪流更新集群中的服務器。
- Assembly.LoadFrom + 強簽名程序集:因為相同標識的程序集在內存中只會加載一次,所以每次功能發生變化,都要增加程序集的版本號。
- Assembly.Load + + 強簽名程序集 + GAC:因為相同標識的程序集在內存中只會加載一次,所以每次功能發生變化,都要增加程序集的版本號。
- Assembly.LoadFile:Assembly.LoadFile 可以多次加載相同標識的程序集,只要程序集所在的目錄位置不同。
重點說一下 Assembly.LoadFile
項目結構
測試代碼
1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Text; 5 using System.Threading.Tasks; 6 using System.Reflection; 7 using System.IO; 8 using Contracts; 9 10 namespace Test 11 { 12 class Program 13 { 14 static void Main(string[] args) 15 { 16 SetupPlugEnvironment(); 17 18 ExecuteOperator("1.0.0.0"); 19 ExecuteOperator("2.0.0.0"); 20 } 21 22 private static void ExecuteOperator(string version) 23 { 24 var operatorType = Type.GetType("Implements.Operator, Implements, version = " + version + ""); 25 var operatorInstance = Activator.CreateInstance(operatorType) as IOperator; 26 operatorInstance.Operate(); 27 } 28 29 private static void SetupPlugEnvironment() 30 { 31 AppDomain.CurrentDomain.AssemblyResolve += CurrentDomain_AssemblyResolve; 32 } 33 34 static Assembly CurrentDomain_AssemblyResolve(object sender, ResolveEventArgs args) 35 { 36 AssemblyName name = new AssemblyName(args.Name); 37 38 var file = Path.Combine( 39 @"E:\Coding\HappyStudy\LoadContextStudy\Test\bin\Debug\Plugs", 40 name.Name, 41 name.Version.ToString(), 42 name.Name + ".dll"); 43 44 Console.WriteLine("加載插件:" + name.Version); 45 46 return Assembly.LoadFile(file); 47 } 48 } 49 }
輸出結果
說明
調用 Type.GetType 會導致 CLR 執行程序集探測過程,在正常的探測路徑下沒有找到程序集就會觸發 AssemblyResolve 事件,為啥會觸發兩次呢?我還不知道,有知道的兄弟請留言。
備注
微軟不推薦使用 LoadFile(會加載相同標識的程序集多次),Add In 采用的是 AppDomain,MEF 采用的是 LoadFrom(我估計是,還沒有看源代碼,測試結果是)。