当前位置: 首页 > 图文教程 > 网络编程 > ASP.NET > 灵活正确的实现.NET插件机制

ASP.NET
asp.net Linq TO Sql 分页方法
asp.net 用XML生成放便扩展的自定义树
asp.ent下合并两个结构相同的DataTable
asp.net 遍历repeater中的控件的几种方式
asp.net 处理原文件中过长的viewstate代码
asp.net下遍历页面中所有的指定控件的代码
获取创建Membership的数据库创建脚本
asp.net AJAX注册类
asp.net 处理F5刷新页面重复提交页面的一个思路
ASP.NET 缓存分析和实践浅析提高运行效率
asp.net 读取并显示excel数据的实现代码
ASP.NET中常用的用来输出JS脚本的类
ASP.NET中内嵌页面代码的一个问题
asp.net(C#)操作excel(上路篇)
一个基于Asp.Net MVC的权限方案
ASP.NET实例教程:51job网站地区选择功能
ASP.NET教程:友好的Html和JS适合SEO
ASP.NET教程:使用.ashx文件去除重复内容
ASP.NET做SEO:制作架构清晰和更新及时的网站地图
ASP.NET优化:Sql注入和Html注入的黑帽SEO

ASP.NET 中的 灵活正确的实现.NET插件机制


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-08-14   浏览: 83 ::
收藏到网摘: n/a

.NET提供的反射(Reflection)机制可以很方便的加载插件。本文提供一种方法,可以灵活的正确的载入所需的插件。

在.NET中,一个完整的类型名称的格式如"类型名,程序集名"。

例如:"System.Configuration.NameValueSectionHandler,System,Version=1.0.3300.0,Culture=neutral,PublicKeyToken=b77a5c561934e089"。

类型名为:System.Configuration.NameValueSectionHandler,这是带名字空间的完整类型名。
你也可以使用该类型的FullName得到。
如:stringtypeName=typeof(NameValueSectionHandler).FullName;
程序集名为:"System,Version=1.0.3300.0,Culture=neutral,PublicKeyToken=b77a5c561934e089",
程序集名为System,系统为自动为其适配扩展名(如System.dll或System.exe);
Version、Culture、PublicKeyToken为程序集的具体版本、文化背景、签名,没有特定要求,这些都可以省略。
我们可以根据类型的名称,来动态载入一个所需要的类型。如:


stringtypeName="System.Configuration.NameValueSectionHandler,System";
Typet=Type.GetType(typeName);
Objectobj=Activator.CreateInstance(t);
//或
System.Configuration.NameValueSectionHandlerobj=(System.Configuration.NameValueSectionHandler)Activator.CreateInstance(t);


此时,obj就是所需要的类型实例。

通常的插件,是需要实现一定的接口的类。因此,在载入插件之前,需要确定该插件类型是否是合适的。

比如,一个插件的接口为IPlugin,那么我们可以用如下方式来识别:

stringinterfaceName=typeof(IPlugin).FullName;
stringtypeName="Muf.MyPlugin,MyPlugin";
Typet=Type.GetType(typeName);

if(t==null
||!t.IsClass
||!t.IsPublic
||t.GetInterface(interfaceName)==null)
{
returnnull;//不是所需要的插件
}


总结上述代码,我们可以做出通用的加载插件的代码:


/**////<summary>
///动态装载并创建类型,该类型拥有指定接口
///</summary>
///<paramname="className">类型名称</param>
///<paramname="interfaceName">指定的接口名称</param>
///<paramname="param">指定构造函数的参数(null或空的数组表示调用默认构造函数)</param>
///<returns>返回所创建的类型(null表示该类型无法创建或找不到)</returns>
publicstaticobjectLoadObject(stringclassName,stringinterfaceName,object[]param)
{
try
{
Typet=Type.GetType(className);

if(t==null
||!t.IsClass
||!t.IsPublic
||t.IsAbstract
||t.GetInterface(interfaceName)==null)
{
returnnull;
}

objecto=Activator.CreateInstance(t,param);
if(o==null)
{
returnnull;
}

returno;
}
catch(Exceptionex)
{
returnnull;
}
}


以后,我们就可以使用LoadObject载入任何所需的插件。

插件一般放在配置文件中,并由程序读入:

配置文件举例(配置文件的使用参见我的相关随笔):

<?xmlversion="1.0"encoding="utf-8"?>
<configuration>
<configSections>
<sectionname="Channels"type="Vmp.Configuration.ChannelsSectionHandler,Communication"/>
</configSections>

<Channels>
<channel
ChannelType="Vmp.Communication.TcpChannel,Communication"
TraceFile="d:\log\channel1.log"
Port="2020"MaxConnections="300"BufferSize="2048"
/>
</Channels>
</configuration>
代码范例:

privateArrayListchannelsList=newArrayList();

privateLoadChannels()
{
ArrayListchannelsConfig=(ArrayList)ConfigurationSettings.GetConfig("Channels");
foreach(HashtableconfiginchannelsConfig)
{
stringchannelType=(string)config["ChannelType"];

IChannelchannel=(IChannel)CommonUtils.LoadObject(channelType,typeof(IChannel).FullName,newobject[]{config});
if(channel==null)
continue;

channelsList.Add(channel);
}

也可以遍历指定的插件目录,并载入所有符合要求的插件,例如:

publicIPlugin[]LoadAllPlugIn(stringpluginDir)
{
//设置默认的插件目录
if(pluginDir==null||pluginDir=="")
pluginDir="./PlugIns";

//获取插件接口名称
stringinterfaceName=typeof(IPlugin).FullName;

//用于存放插件的数组
ArrayListarr=newArrayList();

//遍历插件目录(假设插件为dll文件)
foreach(stringfileinDirectory.GetFiles(pluginDir,"*.dll"))
{
//载入插件文件
Assemblyasm=Assembly.LoadFile(file);
//遍历导出的插件类
foreach(Typetinasm.GetExportedTypes())
{
//载入插件,如果插件不符合指定的接口,则返回null
IPluginplugin=LoadObject(t.FullName,interfaceName,null)asIPlugin;

if(plugin!=null)
arr.Add(plugin);
}
}

//返回插件
return(IPlugin[])arr.ToArray(typeof(IPlugin));
}