.NET:如何實現 “熱插拔”?


背景

如果某個“功能”需要動態更新?這種動態更新,可能是需求驅動的,也可能是為了修改 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(我估計是,還沒有看源代碼,測試結果是)。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM