.NET实现热插拔功能(动态替换功用)方案实例
几种方案
1 脚本化:采用 iron 或 集成其它脚本引擎。
2 appdomain:微软的 add in 框架就是为这个目的设计的。
3 分布式 + 负载平衡 :轮流更新集群中的服务器。
4 assembly.loadfrom + 强签名程序集:因为相同标识的程序集在内存中只会加载一次,所以每次功能发生变化,都要增加程序集的版本号。
5 assembly.load + + 强签名程序集 + gac:因为相同标识的程序集在内存中只会加载一次,所以每次功能发生变化,都要增加程序集的版本号。
6 assembly.loadfile:assembly.loadfile 可以多次加载相同标识的程序集,只要程序集所在的目录位置不同。
重点说一下 assembly.loadfile
测试代码
using system;
using system.collections.generic;
using system.linq;
using system.text;
using system.threading.tasks;
using system.reflection;
using system.io;
using contracts;
namespace test
{
class program
{
static void main(string[] args)
{
setupplugenvironment();
executeoperator("1.0.0.0");
executeoperator("2.0.0.0");
}
private static void executeoperator(string version)
{
var operatortype = type.gettype("implements.operator, implements, version = " + version + "");
var operatorinstance = activator.createinstance(operatortype) as ioperator;
operatorinstance.operate();
}
private static void setupplugenvironment()
{
appdomain.currentdomain.assemblyresolve += currentdomain_assemblyresolve;
}
static assembly currentdomain_assemblyresolve(object sender, resolveeventargs args)
{
assemblyname name = new assemblyname(args.name);
var file = path.combine(
@"e:\coding\happystudy\loadcontextstudy\test\bin\debug\plugs",
name.name,
name.version.tostring(),
name.name + ".dll");
console.writeline("加载插件:" + name.version);
return assembly.loadfile(file);
}
}
}
说明
调用 type.gettype 会导致 clr 执行程序集探测过程,在正常的探测路径下没有找到程序集就会触发 assemblyresolve 事件,为啥会触发两次呢?我还不知道。
微软不推荐使用 loadfile(会加载相同标识的程序集多次),add in 采用的是 appdomain,mef 采用的是 loadfrom。