C# 依賴註入 & MEF

来源:https://www.cnblogs.com/onegarden/archive/2019/02/15/10383056.html
-Advertisement-
Play Games

之前面試有問道依賴註入,因為一直是做客戶端的發開發,沒有接觸這個,後邊工作接觸到了MEF,順便熟悉一下依賴註入 詳細的概念解釋就不講了,網上一大把,個人覺著依賴註入本質是為瞭解耦,方便擴展 依賴註入的方式:屬性註入和構造函數註入,還有介面註入的,看了下跟屬性註入差不多·就不展示了 上代碼: (DI ...


之前面試有問道依賴註入,因為一直是做客戶端的發開發,沒有接觸這個,後邊工作接觸到了MEF,順便熟悉一下依賴註入

詳細的概念解釋就不講了,網上一大把,個人覺著依賴註入本質是為瞭解耦,方便擴展

依賴註入的方式:屬性註入和構造函數註入,還有介面註入的,看了下跟屬性註入差不多·就不展示了

上代碼:

 public interface ICalc
    {
        double Calc(double a, double b);
    }

    public class AddCalc:ICalc
    {
       
        public double Calc(double a, double b)
        {
            return a + b;
        }
    }
    public class SubtractCalc:ICalc
    {
        public double Calc(double a, double b)
        {
            return a - b;
        }
    }

    public class MyClac {

        ICalc _calc;

        //屬性註入
        public ICalc Calc {
            get {
                return _calc;
            }
            set {
                _calc = value;
            }
        }

        //構造函數註入
        public MyClac(ICalc calc)
        {
            _calc = calc;
        }


        public double Calculate(double a, double b)
        {
            return _calc.Calc(a, b);
        }
    }

(DI )依賴註入是實現(IOC)控制反轉的一種方式,但是使用的時候,比如再擴展的時候還是需要修改調用代碼,所以就有了IOC 容器來方便這個調用

.NET 下邊 MEF框架就是乾這個的, 本質是通過特性和反射在運行的時候程式集動態載入。

  //介面聲明

   //最終調用過程介面
    public interface ICalculator
    {
        string Calculate(String input);
    }
    //過程中操作介面
    [InheritedExport]//這裡特性標識子類會被導出,後邊子類可以不用表示export導出特性
    public interface IOperation
    {
        int Operate(int left, int right);
    }
    //這裡定義導出操作名稱,可以用來在導出的操作中進行篩選識別,這個介面不用實現
    public interface IOperationData
    {
        string Symbol { get; }
    }

上邊是介面聲明,下邊實現這些介面

[Export(typeof(IOperation))]
    [ExportMetadata("Symbol", '+')]
    public  class Add : IOperation
    {
        public int Operate(int left, int right)
        {
            return left + right;
        }
    }
    [Export(typeof(IOperation))]
    [ExportMetadata("Symbol", '-')]
    public class Subtract : IOperation
    {

        public int Operate(int left, int right)
        {
            return left - right;
        }
    }
    [Export(typeof(IOperation))]
    [ExportMetadata("Symbol",'/')]
    public class Except : IOperation
    {
        public int Operate(int left, int right)
        {
            return left / right;
        }
    }

    [Export(typeof(ICalculator))]
    class MyCalculator : ICalculator
    {

        [ImportMany(AllowRecomposition = true)]
        IEnumerable<Lazy<IOperation, IOperationData>> operations;

        public string Calculate(string input)
        {
            int left;
            int right;
            char operation;
            int fn = FindFirstNonDigit(input); //finds the operator 
            if (fn < 0) return "Could not parse command.";

            try
            {
                //separate out the operands 
                left = int.Parse(input.Substring(0, fn));
                right = int.Parse(input.Substring(fn + 1));
            }
            catch
            {
                return "Could not parse command.";
            }

            operation = input[fn];

            foreach (Lazy<IOperation, IOperationData> i in operations)
            {
                
                if (i.Metadata.Symbol.Equals( operation))
                    return i.Value.Operate(left, right).ToString();
            }
            return "Operation Not Found!";
        }

        private int FindFirstNonDigit(String s)
        {

            for (int i = 0; i < s.Length; i++)
            {
                if (!(Char.IsDigit(s[i])))
                    return i;
            }
            return -1;
        }
    }

 

這裡因為加了exportmetadata特性,所以繼承類要加上export特性,不然MEF 好像不識別,如果沒有exportmetadata,只需要在介面上邊加上inheritedExport特性就可以了·  MEF會自動導入導出的

這裡是導出,下邊看怎麼導入使用

  private CompositionContainer _container; //這個是容器

        [Import(typeof(ICalculator))]
        public ICalculator calculator; //這個導入的類

        private Program()
        {

            var catalog = new AggregateCatalog();
            catalog.Catalogs.Add(new AssemblyCatalog(typeof(Program).Assembly));//這裡直接導入本程式集內的類
            catalog.Catalogs.Add(new DirectoryCatalog("Extensions", "MEF_Ex.dll"));//這裡導入指定目錄下的DLL,可以設置篩選項或者不設置,把目錄下所有的dll全部導入
           _container = new CompositionContainer(catalog);

            try
            {
                this._container.ComposeParts(this);
                
            }
            catch (CompositionException ex)
            {
                Console.WriteLine(ex.ToString());
            }
            
        }

這裡MEF_Ex.dll是另外一個項目,生成的程式集,放到主程式目錄下Extensions目錄下即可

實現了一個類:

    [Export(typeof(IOperation))]
    [ExportMetadata("Symbol", '%')]
    public class Mod : MEF_Interface.IOperation
    {
        public int Operate(int left, int right)
        {
            return left % right;
        }
    }

在main函數中直接new program即可調用calc的方法

            Program pro = new Program();
            Console.WriteLine(pro.calculator.Calculate("1-2"));

還可以單獨導出類的方法和屬性,以及通過metadata篩選導入的類

完整代碼如下:

    [InheritedExport]
    interface IBookService
    {
        string BookName { get; set; }
        string GetBookName();
    }

   // [Export("MusicBook",typeof(IBookService))]
    class MusicBook : IBookService
    {
        public string BookName { get; set; }

        [Export(typeof(string))]
        public string _publicBookName = "publicBookName";
        [Export(typeof(string))]
        private string _privateBookName = "privateBookName";


        public string GetBookName()
        {
            return "MusicBook";
        }

    }

   // [Export("MusicBook", typeof(IBookService))]
    class MathBook : IBookService
    {
        public string BookName { get; set; }

        [Export(typeof(Func<string>))]
        public string GetBookName()
        {
            return "MathBook";
        }

        [Export(typeof(Func<int,string>))]
        private string privateGetName(int count)
        {
            return $"get {count} MathBook";

        }

    }

   // [Export("MusicBook", typeof(IBookService))]
    class HistoryBook : IBookService
    {
        public string BookName { get; set; }

        public string GetBookName()
        {
            return "HistoryBook";
        }

    }
    [InheritedExport]
    public interface IPlugin
    {
        string Caption { get; }
        void Do();
    }
    public interface IPluginMark
    {
        string Mark { get; }
    }

    [Export(typeof(IPlugin))]
    [ExportMetadata("Mark", "Plugin1")]
   public class Plugin1 : IPlugin
    {
        public string Caption { get { return "Plugin1"; } }
        public void  Do()
        {
            Console.WriteLine("Plugin1 do");
        }
    }
    [Export(typeof(IPlugin))]
    [ExportMetadata("Mark", "Plugin2")]
    public class Plugin2 : IPlugin
    {
        public string Caption { get { return "Plugin2"; } }
        public void Do()
        {
            Console.WriteLine("Plugin2 do");
        }
    }
    [Export(typeof(IPlugin))]
    [ExportMetadata("Mark", "Plugin2")]
    public class Plugin3 : IPlugin
    {
        public string Caption { get { return "Plugin3"; } }
        public void Do()
        {
            Console.WriteLine("Plugin3 do");
        }
    }

    #endregion

    class Program
    {
        #region
        [ImportMany]
        public IEnumerable<IBookService> Services { get; set; }//導入類

        [ImportMany]
        public List<string> InputString { get; set; }//導入屬性

        [Import]
        public Func<string> methodWithoutPara { get; set; }//導入方法
        [Import]
        public Func<int, string> methodWithPara { get; set; }//導入方法

        [ImportMany]
        public IEnumerable< Lazy<IPlugin, IPluginMark>> Plugins { get; set; }

        #endregion

        private CompositionContainer _container;

        [Import(typeof(ICalculator))]
        public ICalculator calculator;

        private Program()
        {

            var catalog = new AggregateCatalog();
            catalog.Catalogs.Add(new AssemblyCatalog(typeof(Program).Assembly));//導出本程式集
            catalog.Catalogs.Add(new DirectoryCatalog("Extensions", "MEF_Ex.dll"));//通過文件導入
           _container = new CompositionContainer(catalog);

            try
            {
                this._container.ComposeParts(this);
                
            }
            catch (CompositionException ex)
            {
                Console.WriteLine(ex.ToString());
            }
            
        }

        static void Main(string[] args)
        {
            Program pro = new Program();
            Console.WriteLine(pro.calculator.Calculate("1-2"));

            var plugins = pro.Plugins;//.Where(v => v.Metadata.Mark == "Plugin2").ToList();//這裡可以做篩選

            foreach (var p in plugins)
            {
                p.Value.Do();
            }

            if (pro.Services != null)
            {
                foreach (var service in pro.Services)
                {
                    Console.WriteLine(service.GetBookName());
                }

                foreach (var str in pro.InputString)
                {
                    Console.WriteLine(str);
                }

                //調用無參數的方法
                if (pro.methodWithoutPara != null)
                {
                    Console.WriteLine(pro.methodWithoutPara());
                }

                //調用有參數的方法
                if (pro.methodWithPara != null)
                {
                    Console.WriteLine(pro.methodWithPara(5));
                }

            }
            Console.ReadLine();
            

        }

   
    }

 

總結:

1  MEF會自動導入對應的類實現,然後自動初始化,但是具體什麼時候初始化以及導入,這裡要註意類的初始化方法 以及是不是有可能多線程的問題以及有依賴

2  導入程式集的方式可以直接導入程式集或者通過文件,看了反編譯的代碼以及.netcore的源碼,底層是使用load 以及loadfrom的方法來時間載入程式集的,所以這玩意理論上應該實現不了熱插拔把·

3  關於.net實現熱插拔,網上有很多玩法,之前有看過通過appdomain 來實現,也就是應用程式域,實現略複雜這裡沒研究,也可以通過load的方式重新載入程式集·但是這些理論上應該做不到所謂的熱插拔吧,起碼程式要重啟把···

4 之前有面試問MEF 怎麼實現熱插拔,直接懵逼了,我是搞清楚。後來想了下,可以換一個方式實現,在MEF基礎上實現AOP,通過aop實現許可權控制,攔截某些操作,或者MEF 載入的時候過濾載入項,這些算熱插拔麽···

 

  

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 原因:python是64位的python,而windll.LoadLibrary只能由32位的python使用 參考: 64位Python調用32位DLL方法(一) 解決方法:使用32位的python(切記版本不要太新,本人一開始使用最新的32位python3.7.2再次報錯,換成python3.6 ...
  • 1.atomic包里的幾個函數以及sync包里的mutex類型,提供瞭解決方案2.原子函數能夠以很底層的加鎖機制來同步訪問整型變數和指針3.atomic.AddInt64(&counter, 1)的原理是強制同一時刻只能有一個goroutine運行並完成這個加法操作 ...
  • 迭代器的執行流程,以及說明可迭代對象不一定是迭代器,但迭代器一定是可迭代對象 實例1 實例1的優化 ...
  • 一、什麼是委托: 委托是定址方法的.NET版本,使用委托可以將方法作為參數進行傳遞。委托是一種特殊類型的對象,其特殊之處在於委托中包含的只是一個活多個方法的地址,而不是數據。 二、使用委托: 關鍵字:delegate 1.聲明: public delegate void DoNothing();// ...
  • 這稱不上一篇技術文。 這邊記錄解決一個問題的過程和感受。這種感覺每個搞IT的人或多或少都感受過,是程式人獨有的快樂之一。只是大部分人沒有將這種感覺記錄下來。但是當你記錄時,這種感覺也早已消失。 需求:通過程式抓取outlook中的尋呼欄位 當然這個需求被我簡化了,實際上這個欄位記錄的是員工的工號。之 ...
  • 一、 打開Visual Studio 2017(我使用的是2017) 新建一個mvc項目 命名為StudentEntity 二、1)建立完項目後在項目中右擊選擇新建項,找到ADO.NET實體數據模型 命名為StudentModel 2)然後選擇來自資料庫的EF設計器(如果你是vs低版本的話可能就只有 ...
  • 簡單闡述 在C 的WinForm裡面,原生控制項是沒有居中屬性的,故通過重寫OnResize(EventArgs e)方法,通過計算,重新定位控制項位置。 以Label控制項為例 (1)將label的AutoSize屬性設置為false;Dock屬性設置為fill;TextAlign屬性設置為Middle ...
  • 出於工作需要,需要製作一個嵌入在桌面應用中的C語言編輯器,經過一系列調研,目前ScintillaNET應該是最合適的了,開源、輕便、功能豐富,但是踩得坑也很多,接下麵一一說道。 目前ScintillaNET托管在https://github.com/jacobslusser/ScintillaNET ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...