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 ...
随机推荐
- CSS--滚动条设置;
CSS滚动条实现步骤及美化小技巧 1.overflow-y : 设置当对象的内容超过其指定高度时如何管理内容:overflow-x : 设置当对象的内容超过其指定宽度时如何管理内容. 参数:visib ...
- 正确答案 全国信息学奥林匹克联赛( ( NOIP2014) 复 赛 模拟题 Day1 长乐一中
[题目描述]小 H 与小 Y 刚刚参加完 UOIP 外卡组的初赛,就迫不及待的跑出考场对答案."吔,我的答案和你都不一样!",小 Y 说道,"我们去找神犇们问答案吧&qu ...
- 数据库实验一 SQL Service的安装
SQL 2014 安装需要 Microsoft .Net Framework 3.5框架 win 8.1 Microsoft .Net Framework 3.5框架安装教程:点我 安装图文教程:点我 ...
- UVALive-7303 Aquarium (最小生成树)
题目大意:在nxm的方格中,每一个1x1的小方格中都有一堵沿对角线的墙,并且每堵墙都有一个坚固程度,这些墙将nxm的方格分割成了若干个区域.现在要拆除一些墙,使其变成一个区域. 题目分析:将区域视作点 ...
- python课程
课程大纲 一.语言基础(5周) 数据类型 流程控制 模块 函数.迭代器.装饰器 递归.迭代.反射 面向对象编程 模拟人生游戏开发 二.网络编程(4周) Socket c/s编程.Twisted网络框架 ...
- ps颜色相加
一. 红+绿=黄 红+蓝=品红 绿+蓝=青 白色:红+绿+蓝=白色.(黄.品红.青,两种以上颜色相加是白色) 互补色:红->青.绿->品红.蓝->黄 二. 品红+黄=红 青+黄=绿 ...
- 让边框和文本一样高,不受line-height影响,可以使用padding,padding可以用于行内元素
如果设置了,display:inline-block;边框大小至少和行高一样.这样就可能比文字高.
- Java的位运算 待整理
位运算:二进制运算 Java的异或运算^ 真^假=真 假^真=真 假^假=假 真^真= 假,这四个是在网上copy的例子,真是1,假是0 但它却是说明了Java异或运算的基本法则,那就是:只要两个条件 ...
- createdb test时报global/pg_filenode.map不存在
实际上是存在的,看到说是ipv6占用了5432的端口号,看了一下的确是,将ipv6关闭重启系统即可. 下面是关闭ipv6的方法: 确认IPV6是否开启 在Linux下确认IPv6是否已经被启用,可以从 ...
- 003. vs2010发布、打包安装程序(转)
本资源来自于网络 1. 在vs2010 选择“新建项目”à“ 其他项目类型”à“ Visual Studio Installerà “安装项目”: 命名为:Setup1 . 这是在VS2010中将有 ...