依賴註入[2]: 基於IoC的設計模式

来源:https://www.cnblogs.com/artech/archive/2018/07/26/net-core-di-02.html
-Advertisement-
Play Games

正如我們在《控制反轉》提到過的,很多人將IoC理解為一種“面向對象的設計模式”,實際上IoC自身不僅與面向對象沒有必然的聯繫,它也算不上是一種設計模式。一般來講,設計模式提供了一種解決某種具體問題的方案,但是IoC既沒有一個針對性的問題領域,其自身沒有提供一種可實施的解決方案,所以我更加傾向於將Io... ...


正如我們在《控制反轉》提到過的,很多人將IoC理解為一種“面向對象的設計模式”,實際上IoC自身不僅與面向對象沒有必然的聯繫,它也算不上是一種設計模式。一般來講,設計模式提供了一種解決某種具體問題的方案,但是IoC既沒有一個針對性的問題領域,其自身沒有提供一種可實施的解決方案,所以我更加傾向於將IoC視為一種設計原則。實際上很多我們熟悉的設計模式背後採用了IoC原則,接下來我們就來介紹幾種典型的“設計模式”。

一、模板方法

提到IoC,很多人首先想到的是DI,但是在我看來與IoC思想最為接近的倒是另一種被稱為“模板方法(Template  Method)”的設計模式。模板方法模式與IoC的意圖可以說不謀而合,該模式主張將一個可復用的工作流程或者由多個步驟組成的演算法定義成模板方法,組成這個流程或者演算法的步驟實現在相應的虛方法之中,模板方法根據按照預先編排的流程去調用這些虛方法。所有這些方法均定義在同一個類中,我們可以通過派生該類並重寫相應的虛方法達到對流程定製的目的。

對於《控制反轉》演示的這個MVC的例子,我們可以將整個請求處理流程實現在如下一個MvcEngine類中,請求的監聽與接收、目標Controller的激活與執行以及View的呈現分別定義在5個受保護的虛方法中,模板方法StartAsync根據預定義的請求處理流程先後調用這5個方法。

public class MvcEngine
{
    public async Task StartAsync(Uri address)
    {
        await ListenAsync(address);
        while (true)
        {
            var request = await ReceiveAsync();
            var controller = await CreateControllerAsync(request);
            var view = await ExecuteControllerAsync(controller);
            await RenderViewAsync(view);
        }
    }
    protected virtual Task ListenAsync(Uri address);
    protected virtual Task<Request> ReceiveAsync();
    protected virtual Task<Controller> CreateControllerAsync(Request request);
    protected virtual Task<View> ExecuteControllerAsync(Controller controller);
    protected virtual Task RenderViewAsync(View view);
}

對於具體的應用來說,如果定義在MvcEngine針對請求的處理方式完全符合它的要求,它只需要創建這個一個MvcEngine對象,然後指定一個對應的基地址調用模板方法StartAsync開啟這個MVC引擎即可。如果該MVC引擎處理請求的某個環節不能滿足它的要求,它可以創建MvcEngine的派生類,並重寫實現該環節的相應虛方法即可。

比如說定義在某個應用程式中的Controller都是無狀態的,它希望採用單例(Singleton)的方式重用已經激活的Controller以提高性能,那麼它就可以按照如下的方式創建一個自定義的FoobarMvcEngine並按照自己的方式重寫

public class FoobarMvcEngine : MvcEngine
{
    protected override Task<View> CreateControllerAsync (Request request)
    {
        <<省略實現>>
    }
}

二、工廠方法

對於一個複雜的流程來說,我們傾向於將組成該流程的各個環節實現在相對獨立的組件之中,那麼針對流程的定製就可以通過提供定製組件的方式來實現。我們知道23種設計模式之中有一種重要的類型,那就是“創建型模式”,比如常用的“工廠方法”和“抽象工廠”,IoC所體現的針對流程的共用與定製可以通過它們來完成。

所謂的工廠方法,說白了就是在某個類中定義用於提供依賴對象的方法,這個方法可以是一個單純的虛方法,也可以是具有預設實現的虛方法,至於方法聲明的返回類型,可以是一個介面或者抽象類,也可以是未被封閉(Sealed)的具體類型。作為它的派生類型,它可以實現或者重寫工廠方法以提供所需的具體對象。

同樣以我們的MVC框架為例,我們讓獨立的組件來完成組成整個請求處理流程的幾個核心環節。具體來說,我們針對這些核心組件定義瞭如下這幾個對應的介面。IWebLister介面用來監聽、接收和響應請求(針對請求的響應由ReceiveAsync方法返回的HttpContext對象來完成,後者表示針對當前請求的上下文),IControllerActivator介面用於根據當前請求激活目標Controller對象,已經在後者執行完成後做一些釋放回收工作。至於IControllerExecutorIViewRender介面則分別用來完成針對Controller的執行和針對View的呈現。

public interface IWebLister
{
    Task ListenAsync(Uri address);
    Task<HttpContext> ReceiveAsync();
}

public interface IControllerActivator
{
    Task<Controller> CreateControllerAsync(HttpContext httpContext);
    Task ReleaseAsync(Controller controller);
}

public interface IControllerExecutor
{
    Task<View> ExecuteAsync(Controller controller, HttpContext httpContext);
}

public interface IViewRender
{
    Task RendAsync(View view, HttpContext httpContext);
}

我們在作為MVC引擎的MvcEngine類中定義了四個工廠方法(GetWebListener、GetControllerActivator、GetControllerExecutor和GetViewRenderer)來提供上述這4種組件。這四個工廠方法均為具有預設實現的虛方法,我們可以利用它們提供預設的組件。在用於啟動引擎的StartAsync方法中,我們利用這些工廠方法提供的對象來具體完成請求處理流程的各個核心環節。

 
public class MvcEngine
{
    public async Task StartAsync(Uri address)
    {
        var listener = GetWebLister();
        var activator = GetControllerActivator();
        var executor = GetControllerExecutor();
        var render = GetViewRender();
        await listener.ListenAsync(address);
        while (true)
        {
            var httpContext = await listener.ReceiveAsync();
            var controller = await activator.CreateControllerAsync(httpContext);
            try
            {
                var view = await executor.ExecuteAsync(controller, httpContext);
                await render.RendAsync(view, httpContext);
            }
            finally

            {
                await activator.ReleaseAsync(controller);
            }
        }
    }
    protected virtual IWebLister GetWebLister(); 
    protected virtual IControllerActivator GetControllerActivator();
    protected virtual IControllerExecutor GetControllerExecutor();
    protected virtual IViewRender GetViewRender();
}

對於具體的應用程式來說,如果需要對請求處理的某個環節進行定製,它需要將定製的操作實現在對應介面的實現類中。在MvcEngine的派生類中,我們需要重寫對應的工廠方法來提供被定製的對象。 比如上面提及的以單例模式提供目標Controller對象的實現就定義在SingletonControllerActivator類中,我們在派生於MvcEngine的FoobarMvcEngine類中重寫了工廠方法GetControllerActivator使其返回一個SingletonControllerActivator對象。

public class SingletonControllerActivator : IControllerActivator
{         
    public Task<Controller> CreateControllerAsync(HttpContext httpContext)
    {
        <<省略實現>>
    }
    public Task ReleaseAsync(Controller controller) => Task.CompletedTask;
}

public class FoobarMvcEngine : MvcEngine
{
    protected override ControllerActivator GetControllerActivator() => new SingletonControllerActivator();
}

三、抽象工廠

雖然工廠方法和抽象工廠均提供了一個“生產”對象實例的工廠,但是兩者在設計上卻有本質的不同。工廠方法利用定義在某個類型的抽象方法或者虛方法實現了針對單一對象提供方式的抽象,而抽象工廠則利用一個獨立的介面或者抽象類來提供一組相關的對象。

具體來說,我們需要定義一個獨立的工廠介面或者抽象工廠類,併在其中定義多個的工廠方法來提供“同一系列”的多個相關對象。如果希望抽象工廠具有一組預設的“產出”,我們也可以將一個未被封閉的具體類作為抽象工廠,以虛方法形式定義的工廠方法將預設的對象作為返回值。我們根據實際的需要通過實現工廠介面或者繼承抽象工廠類(不一定是抽象類)定義具體工廠類來提供一組定製的系列對象。

現在我們採用抽象工廠模式來改造我們的MVC框架。如下麵的代碼片段所示,我們定義了一個名為IMvcEngineFactory的介面作為抽象工廠,定義在其中定義了四個方法來提供請求監聽和處理過程使用到的4種核心對象。如果MVC提供了針對這四種核心組件的預設實現,我們可以按照如下的方式為這個抽象工廠提供一個預設實現(MvcEngineFactory)。

public interface IMvcEngineFactory
{
    IWebLister GetWebLister();
    IControllerActivator GetControllerActivator();
    IControllerExecutor GetControllerExecutor();
    IViewRender GetViewRender();
}

public class MvcEngineFactory: IMvcEngineFactory
{
    IWebLister GetWebLister();
    IControllerActivator GetControllerActivator();
    IControllerExecutor GetControllerExecutor();
    IViewRender GetViewRender();
}

現在我們採用抽象工廠模式來改造我們的MVC框架。我們在創建MvcEngine對象可以提供一個具體的IMvcEngineFactory對象,如果沒有顯式指定,MvcEngine會使用預設的EngineFactory對象。在用於啟動引擎的StartAsync方法中,MvcEngine利用IMvcEngineFactory來獲取相應的對象協作完整對請求的處理流程。

 
public class MvcEngine
{
    public IMvcEngineFactory EngineFactory { get; }
    public MvcEngine(IMvcEngineFactory engineFactory = null) 
    => EngineFactory = engineFactory??new MvcEngineFactory();
        
    public async Task StartAsync(Uri address)
    {
        var listener = EngineFactory.GetWebLister();
        var activator = EngineFactory.GetControllerActivator();
        var executor = EngineFactory.GetControllerExecutor();
        var render = EngineFactory.GetViewRender();
        await listener.ListenAsync(address);
        while (true)
        {
            var httpContext = await listener.ReceiveAsync();
            var controller = await activator.CreateControllerAsync(httpContext);
            try
            {
                var view = await executor.ExecuteAsync(controller, httpContext);
                await render.RendAsync(view, httpContext);
            }
            finally
            {
                await activator.ReleaseAsync(controller);
            }
        }
    }
        
}

如果具體的應用程式需要採用上面定義的SingletonControllerActivator以單例的模式來激活目標Controller,我們可以按照如下的方式定義一個具體的工廠類FoobarEngineFactory。最終的應用程式將這麼一個FoobarEngineFactory對象作為MvcEngine的EngineFactory。

public class FoobarEngineFactory : EngineFactory
{
    public override ControllerActivator GetControllerActivator()
    {
        return new SingletonControllerActivator();
    }
}

public class App
{
    static void Main(string[] args)
    {
        Uri address = new Uri("http://0.0.0.0:8080/mvcapp");
        MvcEngine engine     = new MvcEngine(new FoobarEngineFactory());
        engine.Start(address);
    }
}

除了上面介紹這三種典型的設計,還有很多其他的設計模式,比如策略模式、觀察者模式等等,它們無一不是採用IoC的設計原則。Martin Fowler在《Inversion of Control 》一文中正是通過觀察者模式來介紹IoC的。我們將在下一篇中對依賴註入模式進行深入講解。

依賴註入[1]: 控制反轉
依賴註入[2]: 基於IoC的設計模式
依賴註入[3]: 依賴註入模式
依賴註入[4]: 創建一個簡易版的DI框架[上篇]
依賴註入[5]: 創建一個簡易版的DI框架[下篇]
依賴註入[6]: .NET Core DI框架[編程體驗]
依賴註入[7]: .NET Core DI框架[服務註冊]
依賴註入[8]: .NET Core DI框架[服務消費]


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

-Advertisement-
Play Games
更多相關文章
  • OpenFlow1.3.3 學習記錄(持續更新) 正在學習OpenFlow1.3,該篇筆記將日常更新,主要內容大致為官方文檔的總結與翻譯。 交換機組件 按照優先順序順序進行包匹配,如果匹配到流表項,則執行流表項中綁定的Instructions;如果沒有匹配到流表項,將根據table miss的配置進行 ...
  • 遇到一種業務場景,前端上傳的文件需要經過java服務轉發至文件服務。期間遇到了原生HttpClient怎麼使用的問題、怎麼把MultipartFile怎麼重新組裝成Http請求發送出去的問題、文件中文名亂碼問題。最後都解決了,先上代碼,再講遇到的坑 特別說明及遇到的坑: 1. 這裡基於tomcat進 ...
  • 大佬的總結(大贊!) final可以修飾:屬性,方法,類,局部變數(方法中的變數) final修飾的屬性的初始化可以在編譯期,也可以在運行期,初始化後不能被改變。 final修飾的屬性跟具體對象有關,在運行期初始化的final屬性,不同對象可以有不同的值。 final修飾的屬性表明是一個常數(創建後 ...
  • 一, 繼承 繼承是一種創建新類的方式,在python中,新建的類可以繼承一個或多個父類,父類又可稱為 基類或超類,新建的類稱為派生類或子類 1. python中類的繼承分為:單繼承和多繼承 2. 查看繼承 提示:如果沒有指定基類,python的類會預設繼承object類,object是所有pytho ...
  • 對象操作流(序列化流) 每次讀取和寫出的都是JavaBean對象. 序列化:將對象寫入到文件中的過程 反序列化:從文件中讀取對象到程式的過程 transient: 標識瞬態,序列化的時候,該修飾符修飾的成員不能序列化 ObjectOutputStream 構造方法: public ObjectOut ...
  • 要想學習java語言,首先要搭建Java的開發環境,包括開發環境和運行環境,那就要下載jdk的安裝包來進行搭建了 下載地址:鏈接: https://pan.baidu.com/s/1msUuHYRfIjxyPnwZEksNRw 密碼: ssv4 首先我們先來瞭解一下JDK JDK:java deve ...
  • """關係型資料庫(SQL資料庫):MySQL、Oracle、Sqlite3、SQLServer...1.可以存儲統一格式的數據2.可以用於保存大量的數據3.表與表之間有關聯 非關係型數據(NoSQL資料庫):Mangodb、Redis....""" # 1.建立資料庫連接# connect() 若 ...
  • 1. 重新排列數列,使得數組左邊為奇數,右邊為偶數 空間複雜度O(1) 時間複雜度O(n) 思路:兩個指針分別指向數組的頭和尾,頭指針正向遍曆數組,找到第一個偶數 尾指針反向遍曆數組,找到第一個奇數,兩者交換 2. 如何找出數組中唯一的重覆元素 每個數組只能訪問一次,不能用輔助空間 數組a[n] 數 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...