.NET框架之---MEF托管可扩展框架

MEF简介:

今天学习了下MEF框架,MEF,全称Managed Extensibility Framework(托管可扩展框架)。MEF是专门致力于解决扩展性问题的框架,MSDN中对MEF有这样一段说明:

  Managed Extensibility Framework 或 MEF 是一个用于创建可扩展的轻型应用程序的库。 应用程序开发人员可利用该库发现并使用扩展,而无需进行配置。 扩展开发人员还可以利用该库轻松地封装代码,避免生成脆弱的硬依赖项。 通过 MEF,不仅可以在应用程序内重用扩展,还可以在应用程序之间重用扩展。


我们通过例子1来看下MEF是如何工作的:


例子1:

新建个WPF程序--MEFtestpro,添加引用System.ComponentModel.Composition,MEF的核心就是在此类库中实现的

项目结构图:


添加一个接口IApple.cs

namespace MEFtestpro
{
  public  interface IApple
    {
        string GetAppleColor();
    }
}


然后添加三个类RedApple.cs

namespace MEFtestpro
{
    [Export ("Apple",typeof(IApple))]     //将RedApple类导出类型为IApple接口
    class RedApple : IApple
    {
        public string GetAppleColor()
        {
            return "Red";
        }
    }

    [Export("Apple", typeof(IApple))]     //将GreedApple类导出类型为IApple接口
    class GreedApple : IApple
    {
        public string GetAppleColor()
        {
            return "Green";
        }
    }

    [Export("Apple", typeof(IApple))]     //将YellowApple类导出类型为IApple接口
    class YellowApple : IApple
    {
        public string GetAppleColor()
        {
            return "Yellow";
        }
    }
}


最后,主程序MainWindow.xaml.cs

namespace MEFtestpro
{
    public partial class MainWindow : Window
    {
        [ImportMany("Apple")]       //Apple是契约名字,可以任意起,但是要注意别重名
        public IEnumerable<IApple> Apples { get; set; }
        public MainWindow()
        {
            InitializeComponent();
            this.Compose();
            if (this.Apples != null)
            {
                string s=string.Empty;
                foreach (var apple in Apples)   //将内容显示在label上
                {
                    s=s+ apple.GetAppleColor()+"
";  
                    label.Content = s;
                }
            }
        }

        //这个方法表示添加当前Program这个类到组合容器,为什么要添加到组合容器?
        //是因为只要添加到组合容器中之后,如果该类里面有Import,MEF才会自动去寻找对应的Export。
        //这也就是为什么使用MEF前必须要组合部件的原因。
        private void Compose()
        {
            var catalog = new AssemblyCatalog(Assembly.GetExecutingAssembly());
            CompositionContainer container = new CompositionContainer(catalog);
            container.ComposeParts(this);
        }
    }
}



然后运行看结果,如下:


可以看到,我们没有实例化类,而是仅仅通过Export和Import一个中间类型(IApple),就实现了调用类中的方法!


这就实现了主程序和类之间的解耦,大大提高了代码的扩展性和易维护性!

可能有人就会说多此一举,既然我们可以实例化类,为什么非要用这种奇怪的语法。

其实如果我们站在软件框架设计的层面,它的好处就是可以减少dll之间的引用,使你的程序更加健壮可扩展


接下来请看例子2.



例子2:

新建控制台程序MEFtestPro2


项目结构图:




新增.NET类库Fruit,里面包含了一个接口IFruit

namespace Fruit
{
    public interface IFruit
    {
        string GetFruitName();
    }
}


新增.NET类库Banana,引用Fruit.DLL

namespace Banana
{
    [Export(typeof(IFruit))]
    public class Banana : IFruit
    {
        public string GetFruitName()
        {
            return "Banana";
        }
    }
}

新增.NET类库Orange,引用Fruit.DLL

namespace Orange
{
     [Export(typeof(IFruit))]
    public class Orange : IFruit
    {
        public string GetFruitName()
        {
            return "Orange";
        }
    }
}


最后主程序program.cs,引用Fruit.DLL

namespace MEFtestPro2
{
    class Program
    {
        [ImportMany(typeof(IFruit))]  
        public IEnumerable<IFruit> fruits { get; set; }
        static void Main(string[] args)
        {
            Program pro = new Program();
            pro.Compose();
            foreach (var f in pro.fruits) //打印输出
            {
                Console.WriteLine(f.GetFruitName());
            }
            Console.Read();
        }

        private void Compose()
        {
            var catalog = new DirectoryCatalog("fruits"); //fruits是一个目录名称,就是主程序所在目录(bin-Debug-fruits)文件夹(我们需要提前建立好)
            var container = new CompositionContainer(catalog);
            container.ComposeParts(this);
        }
    }
}


现在,我们把生成的Banana.dll和Orange.dll拷贝到这个文件夹下,如图:



然后运行才可以正确输出信息(毕竟我们没有引用那个项目),如图:


注意,我们仅仅是把Banana.dll和Orange.dll拷贝到我们指定的目录,然后通过MEF导入导出中间类型(IFruit),

实现了主程序调用未知的DLL中的方法,而主程序并未引用该DLL



总结一下MEF框架的好处:

1.解耦。试想下,如果主程序引用了Banana.dll和Orange.dll,那么就意味着你可以无限制的开放DLL中类,属性,方法的访问权限,也就意味着主程序中会出现很多耦合的代码,哪天你想移除这个DLL,程序肯定编译失败,而且你要手动删除这些耦合代码,而MEF因为是通过中间接口来完成调用的,所以只向外暴露了接口里面的成员,程序员是无法任意调用DLL中的任何方法,只能通过接口来调用。就算删掉这个DLL,程序也能正常运行!


2.可扩展性。举个例子,假设你的程序已经移交给客户了,哪天客户说我不想看Banana了,我想换一个水果,苹果Apple,这时,你只需重写一个Apple.DLL,使其继承并实现IFruit接口,然后,只要将Apple.dll交给客户,并让其覆盖Banana.DLL,打开程序,你会发现香蕉变成了苹果。是不是很方便!如果是以前,你可能得重新将所有有关banana的东西全部替换,然后重新编译,发布,再将整个程序移交客户,这样说大家应该都明白了!


3.MEF不仅可以导出类,还可以导出方法,属性,不管是私有还是公有,从而满足更多的需求!


最后,有人需要上述的源程序例子,可点击链接下载:http://download.csdn.net/download/wcc27857285/10030752


参考资料:http://www.cnblogs.com/yk123/p/5350133.html


未完待续。。。。。

原文地址:https://www.cnblogs.com/kevinWu7/p/10163518.html