MEF简单示例
原文地址: http://www.cnblogs.com/xiaokang088/archive/2012/02/21/2361631.html
MEF 的精髓在于插件式开发,方便扩展。
例如,应用程序的日志输出到文本,后来想改为输出到数据库,按照传统的办法,需要替换项目,删除原有的引用,增加新的引用;如果使用MEF,直接用新的dll替换原来的dll,即可搞定,这就是MEF的魅力。
下面就用简单的例子来实现上述的需求。
1. 建立一个解决方案,然后增加如下的几个项目
DBlog 输出日志到数据库的实现
TextLog 输出日志到文本的实现
ILog 输出日志的接口,调用方和实现者的中间桥梁
MEFConsoleHost 控制台应用程序,模拟实用场合
MEFWPFHost WPF 应用程序,模拟实用场合
2. 先定义接口ILog,非常简单,就一个方法
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text; namespace ILog
{
public interface ILogService
{
void Log(string content);
}
}
3. 输出到文本的实现TextLog
首先添加引用:引用刚才添加的接口Ilog 和System.ComponentModel.Composition
然后增加类TextLogService.cs,继承接口,并实现方法。
注意 类和方法都Public。
最后定义导出[Export(typeof(ILog.ILogService))]
这个是最主要的,和普通的类库区别也在这里
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Diagnostics;
using System.ComponentModel.Composition; namespace TextLog
{
[Export(typeof(ILog.ILogService))]
public class TextLogService : ILog.ILogService
{
public void Log(string content)
{
Console.WriteLine("TextLog:" + content); System.Diagnostics.TextWriterTraceListener TraceListener = new System.Diagnostics.TextWriterTraceListener(@"d:/debug.txt");
System.Diagnostics.Debug.Listeners.Add(TraceListener);
System.Diagnostics.Debug.Write(System.DateTime.Now.ToString()+" ");
Debug.Write("TextLog:" + content);
Debug.WriteLine("~~");
TraceListener.Flush();
}
}
}
4. 输出到数据库的实现DBLog,实现方法同上例,输出日志的时候区分一下。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Diagnostics;
using System.ComponentModel.Composition; namespace DBLog
{
[Export(typeof(ILog.ILogService))]
public class DBLogService: ILog.ILogService
{
public void Log(string content)
{
Console.WriteLine("DBLog:" + content); System.Diagnostics.TextWriterTraceListener TraceListener = new System.Diagnostics.TextWriterTraceListener(@"d:/debug.txt");
System.Diagnostics.Debug.Listeners.Add(TraceListener);
System.Diagnostics.Debug.Write(System.DateTime.Now.ToString()+" ");
Debug.Write("DBLog:" + content);
Debug.WriteLine("");
TraceListener.Flush();
}
}
}
5. 调用方MEFConsoleHost, 这个也需要增加两个引用,
ILog 和System.ComponentModel.Composition
主程序代码如下:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.ComponentModel.Composition;
using System.ComponentModel.Composition.Hosting; namespace MEFConsoleHost
{
class Program
{
[Import(typeof(ILog.ILogService))]
public ILog.ILogService CurrentLogService { get; set; } static void Main(string[] args)
{
Program pro = new Program();
pro.run();
} private void run()
{
//var catalog = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory,"DbLog.dll");
var catalog = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory, "TextLog.dll"); var container = new CompositionContainer(catalog);
container.ComposeParts(this); CurrentLogService.Log("*MEF Log Test Pass*");
Console.Read();
}
}
}
注意两个地方
a. 导入声明
[Import(typeof(ILogService))]
public ILogService CurrentLogService { get; set; }
用接口来定义实例,然后增加导入声明,和导出的相互对应
b. 建立Catalog和Container
var catalog = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory, "TextLog.dll");
var container = new CompositionContainer(catalog);
container.ComposeParts(this);
6. 把所有项目的输出都指定到同一个目录,也就说让dll和exe在同一目录,运行控制台程序,输出
TextLog: MEF Log Test Pass
7.如果我们要输出到数据库,把上面的catalog那一句改成
var catalog = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory,"DBLog.dll");
如果不想这么麻烦,这个目录里面只放DBLog.dll 或者TextLog.dll ,然后把上面那句改为:
var catalog = new DirectoryCatalog(AppDomain.CurrentDomain.BaseDirectory);
照样可以用。
如果两个都在,还这么搞,那不行的,因为就定义了一个实例,同目录有两个dll。
MEF简单示例的更多相关文章
- Linux下的C Socket编程 -- server端的简单示例
Linux下的C Socket编程(三) server端的简单示例 经过前面的client端的学习,我们已经知道了如何创建socket,所以接下来就是去绑定他到具体的一个端口上面去. 绑定socket ...
- C# 构建XML(简单示例)
C# 构建XML的简单示例: var pars = new Dictionary<string, string> { {"url","https://www. ...
- 根据juery CSS点击一个标签弹出一个遮罩层的简单示例
<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...
- ACEXML解析XML文件——简单示例程序
掌握了ACMXML库解析XML文件的方法后,下面来实现一个比较完整的程序. 定义基本结构 xml文件格式如下 <?xml version="1.0"?> <roo ...
- demo工程的清单文件及activity中api代码简单示例
第一步注册一个账户,并创建一个应用.获取app ID与 app Key. 第二步下载sdk 第三步新建工程,修改清单文件,导入相关的sdk文件及调用相应的api搞定. 3.1 修改清单文件,主要是加入 ...
- spring-servlet.xml简单示例
spring-servlet.xml简单示例 某个项目中的spring-servlet.xml 记下来以后研究用 <!-- springMVC简单配置 --> <?xml versi ...
- SignalR 简单示例
一.什么是 SignalR ASP.NET SignalR is a library for ASP.NET developers that simplifies the process of add ...
- Web API 简单示例
一.RESTful和Web API Representational State Transfer (REST) is a software architecture style consisting ...
- XML引入多scheme文件约束简单示例
XML引入多scheme文件约束简单示例,用company.xsd和department.xsd来约束company.xml: company.xsd <?xml version="1 ...
随机推荐
- linux中socket的理解---4
一.socket 一般来说socket有一个别名也叫做套接字. socket起源于Unix,都可以用“打 开open –> 读写write/read –> 关闭close”模式来操作.So ...
- C/C++程序员必须熟练应用的开源项目(转-清风小阁)
作为一个经验丰富的C/C++程序员, 肯定亲手写过各种功能的代码, 比如封装过数据库访问的类, 封装过网络通信的类,封装过日志操作的类, 封装过文件访问的类, 封装过UI界面库等, 也在实际的项目中应 ...
- Java泛型和链表
泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型接口.泛型方法. Java语言引 ...
- 错误:无法访问android.app.Activity 找不到android.app.Activity的类文件
视频里面在工程ndk22/bin/classes中 运行javah com.cn.ndk22.Ndk22.Activity ,出现了.h文件 但是我在bin/classes目录中运行javah 时出 ...
- 三星Mega 6.3(i9200)删除kingroot
关于kingroot实际效果怎么样,是不是流氓软件,我不做评论. 手机型号,三星galaxy mega 6.3(i9200)水货,更详细的机型不知道怎么看. 刷了官方的rom以后,下载了kingroo ...
- 黑马程序员——JAVA基础之IO流FileReader,FileWriter
------- android培训.java培训.期待与您交流! ---------- IO(Input Output)流 IO流用来处理设备之间的数据传输 Java对数据的操作是通过流的方式 J ...
- -bash: crontab: command not found(转)
操作步骤 1. 确认crontab是否安装: 执行 crontab 命令如果报 command not found,就表明没有安装 2. 安装 crontab 执行 yum install -y vi ...
- C#代码:用事件模式实现通知
事件提供了一种标准的机制来通知监听者..NET的事件模式使用了事件语法来实现观察者模式.任意数量的客户对象都可以将自己的处理函数注册到事件上,然后处理这些事件.这些客户对象不需要再编译期就给出.时间也 ...
- Spring MVC静态资源处理——<mvc:resources /> ||<mvc:default-servlet-handler /> 转载
Spring MVC静态资源处理——<mvc:resources /> ||<mvc:default-servlet-handler /> mvcmvc:resources ...
- asp.net系统过滤器、自定义过滤器
原文地址:http://www.cnblogs.com/kissdodog/archive/2013/05/21/3090513.html 一.系统过滤器使用说明 1.OutputCache过滤器 O ...