.NET:如何实现 “热插拔”?

背景

如果某个“功能”需要动态更新?这种动态更新,可能是需求驱动的,也可能是为了修改 BUG,面对这种场景,如何实现“热插拔”呢?先解释一下“热插拔”:在系统运行过程动态替换某些功能,不用重启系统进程。

几种方案

  • 脚本化:采用 Iron 或 集成其它脚本引擎。
  • AppDomain:微软的 Add In 框架就是为这个目的设计的。
  • 分布式 + 负载平衡 :轮流更新集群中的服务器。
  • Assembly.LoadFrom + 强签名程序集:因为相同标识的程序集在内存中只会加载一次,所以每次功能发生变化,都要增加程序集的版本号。
  • Assembly.Load +  + 强签名程序集 + GAC:因为相同标识的程序集在内存中只会加载一次,所以每次功能发生变化,都要增加程序集的版本号。
  • Assembly.LoadFile:Assembly.LoadFile 可以多次加载相同标识的程序集,只要程序集所在的目录位置不同。

重点说一下 Assembly.LoadFile

项目结构

.NET:如何实现 “热插拔”?

测试代码

 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 }

输出结果

.NET:如何实现 “热插拔”?

说明

调用 Type.GetType 会导致 CLR 执行程序集探测过程,在正常的探测路径下没有找到程序集就会触发 AssemblyResolve 事件,为啥会触发两次呢?我还不知道,有知道的兄弟请留言。

备注

微软不推荐使用 LoadFile(会加载相同标识的程序集多次),Add In 采用的是 AppDomain,MEF 采用的是 LoadFrom(我估计是,还没有看源代码,测试结果是)。

 

你可能感兴趣的:(.net)