zoukankan      html  css  js  c++  java
  • .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


    未完待续。。。。。

  • 相关阅读:
    lua的数组下标是从1开始的
    DestroyImmediate的一些坑
    c#的IDisposable
    unity工具开发(转)
    winform控件命名规范对照表
    C#调用Exe程序示例
    System.Diagnostics.Process.Start的妙用
    C#中AppDomain.CurrentDomain.BaseDirectory及各种路径获取方法
    C# WindowsAPI
    TabPage判断重复添加Page
  • 原文地址:https://www.cnblogs.com/kevinWu7/p/10163518.html
Copyright © 2011-2022 走看看