使用 C# (.NET Core) 實現模板方法模式 (Template Method Pattern)

来源:https://www.cnblogs.com/cgzl/archive/2018/04/17/8865861.html
-Advertisement-
Play Games

本文的概念內容來自深入淺出設計模式一書. 項目需求 有一家咖啡店, 供應咖啡和茶, 它們的工序如下: 咖啡: 茶: 可以看到咖啡和茶的製作工序是差不多的, 都是有4步, 其中有兩步它們兩個是一樣的, 另外兩步雖然具體內容不一樣, 但是都做做的同一類工作. 現在問題也有了, 當前的設計兩個類裡面有很多 ...


本文的概念內容來自深入淺出設計模式一書.

項目需求

有一家咖啡店, 供應咖啡和茶, 它們的工序如下:

咖啡:

茶:

可以看到咖啡和茶的製作工序是差不多的, 都是有4步, 其中有兩步它們兩個是一樣的, 另外兩步雖然具體內容不一樣, 但是都做做的同一類工作.

現在問題也有了, 當前的設計兩個類裡面有很多重覆的代碼, 那麼應該怎樣設計以減少冗餘呢?

初次嘗試

把共有的方法放到父類裡面, 把不同的方法放到子類裡面.

父類裡面有一個抽象的prepareRecipe()方法[翻譯為準備烹飪方法/製作方法], 然後在不同的子類裡面有不同的實現. 也就是說每個子類都有自己製作飲料的方法.

再仔細想想應該怎樣設計

可以發現兩個飲料的製作方法遵循了同樣的演算法:

  1. 把水燒開
  2. 用開水沖咖啡或茶
  3. 把沖開的飲料放到杯里
  4. 添加適當的調料

現在我們來抽像prepareRecipe()方法:

1.先看看兩個飲料的差異:

兩種飲料都有四道工序, 兩個是完全一樣的, 另外兩個在具體的實現上是略有不同的, 但是還是同樣性質的工序.

這兩道不同的工序的本質就是沖飲料和添加調料, 所以prepareRecipe()可以這樣寫:

2. 把上面的方法放到超類里:

這個父類是抽象的, prepareRecipe()將會用來製作咖啡或者茶, 而且我不想讓子類去重寫這個方法, 因為製作工序(演算法)是一定的.

只不過裡面的第2部和第4部是需要子類自己來實現的. 所以brew()和addCondiments()是兩個抽象的方法, 而另外兩個方法則直接在父類裡面實現了.

3. 最後茶和咖啡就是這個樣子的:

 

我們做了什麼?

我們意識到兩種飲料的工序大體是一致的, 儘管某些工序需要不同的實現方法. 所以我們把這些飲料的製作方法歸納到了一個基類CaffeineBeverage裡面.

CaffeineBeverage控制著整個工序, 第1, 3部由它自己完成, 第2, 4步則是由具體的飲料子類來完成.

初識模板方法模式

上面的需求種, prepareRecipe() 就是模板方法. 因為, 它首先是一個方法, 然後它還充當了演算法模板的角色, 這個需求里, 演算法就是製作飲料的整個工序.

所以說: 模板方法定義了一個演算法的步驟, 並允許子類提供其中若幹個步驟的具體實現.

捋一遍整個流程

1. 我需要做一個茶:

2. 然後調用茶的模板方法:

3. 在模板方法裡面執行下列工序:

boildWater();

brew();

pourInCup();

addCondiments();

模板方法有什麼好處?

不使用模板方法時:

  • 咖啡和茶各自控制自己的演算法.
  • 飲料間的代碼重覆.
  • 改變演算法需要修改多個地方
  • 添加新飲料需要做很多工作.
  • 演算法分佈在了不同的類裡面

使用模板方法後:

  • CaffeineBeverage這個父類控制並保護演算法
  • 父類最大化的代碼的復用
  • 演算法只在一個地方, 改變演算法也只需改變這個地方
  • 新的飲料只需實現部分工序即可
  • 父類掌握著演算法, 但是依靠子類去做具體的實現.

模板方法定義

模板方法在一個方法里定義了一套演算法的骨架, 演算法的某些步驟可以讓子類來實現. 模板方法讓子類重新定義演算法的某些步驟而無需改變演算法的結構.

類圖:

這個抽象類:

針對這個抽象類, 我們可以有一些擴展:

看這個hook方法, 它是一個具體的方法, 但是啥也沒做, 這種就叫做鉤子方法. 子類可以重寫該方法, 也可以不重寫.

模板方法裡面的鉤子

所謂的鉤子, 它是一個在抽象類裡面聲明的方法, 但是方法裡面預設的實現是空的. 這也就給了子類"鉤進"演算法某個點的能力, 當然子類也可以不這麼做, 就看子類是否需要了.

看這個帶鉤子的飲料父類:

customerWantsCondiments()就是鉤子, 子類可以重寫它.

在prepareRecipe()方法裡面, 通過這個鉤子方法的結果來決定是否添加調料.

下麵是使用這個鉤子的咖啡:

C#代碼實現

不帶鉤子的父類:

using System;

namespace TemplateMethodPattern.Abstractions
{
    public abstract class CaffeineBeverage
    {
        public void PrepareRecipe()
        {
            BoilWater();
            Brew();
            PourInCup();
            AddCondiments();
        }

        protected void BoilWater()
        {
            Console.WriteLine("Boiling water");
        }

        protected abstract void Brew();

        protected void PourInCup()
        {
            Console.WriteLine("Pouring into cup");
        }

        protected abstract void AddCondiments();
    }
}

咖啡和茶:

using System;
using TemplateMethodPattern.Abstractions;

namespace TemplateMethodPattern.Beverages
{
    public class Coffee: CaffeineBeverage
    {
        protected override void Brew()
        {
            Console.WriteLine("Dripping Coffee through filter");
        }

        protected override void AddCondiments()
        {
            Console.WriteLine("Adding Sugar and Milk");
        }
    }
}

using System;
using TemplateMethodPattern.Abstractions;

namespace TemplateMethodPattern.Beverages
{
    public class Tea: CaffeineBeverage
    {
        protected override void Brew()
        {
            Console.WriteLine("Steeping the tea");
        }

        protected override void AddCondiments()
        {
            Console.WriteLine("Adding Lemon");
        }
    }
}

測試:

var tea = new Tea();
tea.PrepareRecipe();

 

帶鉤子的父類:

using System;

namespace TemplateMethodPattern.Abstractions
{
    public abstract class CaffeineBeverageWithHook
    {
        public void PrepareRecipe()
        {
            BoilWater();
            Brew();
            PourInCup();
            if (CustomerWantsCondiments())
            {
                AddCondiments();
            }
        }

        protected abstract void Brew();
        protected abstract void AddCondiments();

        protected void BoilWater()
        {
            Console.WriteLine("Boiling water");
        }

        protected void PourInCup()
        {
            Console.WriteLine("Pouring into cup");
        }

        public virtual bool CustomerWantsCondiments()
        {
            return true;
        }
    }
}

咖啡:

using System;
using TemplateMethodPattern.Abstractions;

namespace TemplateMethodPattern.Beverages
{
    public class CoffeeWithHook: CaffeineBeverageWithHook
    {
        protected override void Brew()
        {
            Console.WriteLine("Dripping Coffee through filter");
        }

        protected override void AddCondiments()
        {
            Console.WriteLine("Adding Sugar and Milk");
        }

        public override bool CustomerWantsCondiments()
        {
            var answer = GetUserInput();
            if (answer == "yes")
            {
                return true;
            }
            return false;
        }

        private string GetUserInput()
        {
            Console.WriteLine("Would you like milk and sugar with you coffee (y/n) ?");
            var keyInfo = Console.ReadKey();
            return keyInfo.KeyChar == 'y' ? "yes" : "no";
        }
    }
}

測試:

        static void MakeCoffeeWithHook()
        {
            var coffeeWithHook = new CoffeeWithHook();
            Console.WriteLine("Making coffee...");
            coffeeWithHook.PrepareRecipe();
        }

鉤子和抽象方法的區別?

抽象方法是演算法裡面必須要實現的一個方法或步驟, 而鉤子是可選實現的.

 

好萊塢設計原則

好萊塢設計原則就是: 別給我們打電話, 我們會給你打電話.

好萊塢原則可以防止依賴關係腐爛. 依賴關係腐爛是指高級別的組件依賴於低級別的組件, 它又依賴於高級別組件, 它又依賴於橫向組件, 又依賴於低級別組件....以此類推. 當腐爛發生的時候, 沒人會看懂你的系統是怎麼設計的.

而使用好萊塢原則, 我們可以讓低級別組件鉤進一個系統, 但是高級別組件決定何時並且以哪種方式它們才會被需要. 換句話說就是, 高級別組件對低級別組件說: "別給我們打電話, 我們給你們打電話".

好萊塢原則和模板方法模式

模板方法里, 父類控制演算法, 併在需要的時候調用子類的方法.

而子類從來不會直接主動調用父類的方法.

其他問題

好萊塢原則和依賴反轉原則DIP的的區別?

DIP告訴我們不要使用具體的類, 儘量使用抽象類. 而好萊塢原則則是讓低級別組件可以被鉤進演算法中去, 也沒有建立低級別組件和高級別組件間的依賴關係.

三種模式比較:

模板方法模式: 子類決定如何實現演算法中特定的步驟

策略模式: 封裝變化的行為並使用委托來決定哪個行為被使用.

工廠方法模式: 子類決定實例化哪個具體的類.

使用模板方法做排序

看看java裡面數組的排序方法:

mergeSort就可以看做事模板方法, compareTo()就是需要具體實現的方法.

但是這個並沒有使用子類, 但是根據實際情況, 還是可以靈活使用的, 你需要做的就是實現Comparable介面即可., 這個介面裡面只有一個CompareTo()方法.

具體使用C#就是這樣:

鴨子:

using System;

namespace TemplateMethodPattern.ForArraySort
{
    public class Duck : IComparable
    {
        private readonly string _name;
        private readonly int _weight;

        public Duck(string name, int weight)
        {
            _name = name;
            _weight = weight;
        }

        public override string ToString()
        {
            return $"{_name} weights {_weight}";
        }

        public int CompareTo(object obj)
        {
            if (obj is Duck otherDuck)
            {
                if (_weight < otherDuck._weight)
                {
                    return -1;
                }
                if (_weight == otherDuck._weight)
                {
                    return 0;
                }
            }
            return 1;
        }
    }
}

比較鴨子:

        static void SortDuck()
        {
            var ducks = new Duck[]
            {
                new Duck("Duffy", 8),
                new Duck("Dewey",  2),
                new Duck("Howard", 7),
                new Duck("Louie", 2),
                new Duck("Donal", 10),
                new Duck("Huey", 3)
            };
            Console.WriteLine("Before sorting:");
            DisplayDucks(ducks);

            Array.Sort(ducks);

            Console.WriteLine();
            Console.WriteLine("After sorting:");
            DisplayDucks(ducks);
        }

        private static void DisplayDucks(Duck[] ducks)
        {
            foreach (Duck t in ducks)
            {
                Console.WriteLine(t);
            }
        }

效果:

其他鉤子例子

java的JFrame:

JFrame父類裡面有一個update()方法, 它控制著演算法, 我們可以使用paint()方法來鉤進到該演算法的那部分.

父類裡面JFrame的paint()啥也沒做, 就是個鉤子, 我們可以在子類裡面重寫paint(), 上面例子的效果就是:

 

另一個例子Applet小程式:

 

這5個方法全是重寫的鉤子...

我沒看過winform或者wpf/sl的源碼, 我估計也應該有一些鉤子吧.

總結

好萊塢原則: "別給我們打電話, 我們給你打電話"

模板方法模式: 模板方法在一個方法里定義了一套演算法的骨架, 演算法的某些步驟可以讓子類來實現. 模板方法讓子類重新定義演算法的某些步驟而無需改變演算法的結構

該系列的源碼: https://github.com/solenovex/Head-First-Design-Patterns-in-CSharp


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

-Advertisement-
Play Games
更多相關文章
  • 如果一個對象的值為null,那麼它調用擴展方法時會報錯嗎? 上述代碼出現的情況不會報錯,剛開始遇到這種情況時很納悶,就去問了大牛。大牛解釋如下: 擴展函數其實只是為了讓代碼更具有可讀性, 但最終在clr中會翻譯成標準的靜態函數調用, 比如: 調用 "string".ExtMethod()最終會翻譯成 ...
  • ASP.NET Core是微軟新推出支持跨平臺、高性能、開源的開發框架,相比起原有的ASP.NET來說,ASP.NET Core更適合開發現代應用程式,如跨平臺、Dorker的支持、集成現代前端開發框架(如npm、bower、gulp等等)。另外相比ASP.NET它的性能更好,還內置了依賴註入等功能 ...
  • 不管我們學習什麼語言,一開始都是語法,對於面向對象的語言來講,學習完語法之後,就是OOP了,主要還是三大概念:繼承,多態,封裝。而且我們經常也會遇到一些面試題,會考察我們父子類之間的繼承關係等。 這段時間深造ASP.NET-MVC框架,研讀<<asp.net-mvc框架揭秘>>一書的時候,感覺到了自 ...
  • 設計說明 由圖可知: 1.我們需要列印出九行; 2.每行中最大列數等於行數; 代碼實現 效果圖 如果具有強迫症(例如:我^_^)就會發現在第三行和第四行與下麵的行並未對齊; 看效果圖我們會發現,只需要在3*2與4*2的結果後多列印一個空格,乘法表都將對齊; 代碼實現 效果圖 完整代碼 ...
  • protected void btn_Cisco_Click(object sender, EventArgs e) { try { string ip = txt_ip.Value; string community = txt_ttm.Value; string oid = txt_oid.Va ...
  • 繼續上篇的內容,本篇來學習下nginx的配置和守護進程supervisor的使用。 一、Nginx安裝及配置 (1)安裝nginx (2)啟動nginx (3)配置防火牆 成功完成以上配置後,測試nginx是否可以訪問,本地瀏覽器輸入127.0.0.1,成功顯示如下 (4)修改配置文件轉發.net ...
  • 最近在研究.net Core,因為公司的項目用到的都是Oracle資料庫,所以簡單試一下.net Core怎樣連接Oracle。 Oracle官方現在已經提供.net Core的官方驅動(預覽版),也可以通過NuGet直接下載(推薦),下麵來看具體步驟 首先使用visual studio 2017 ...
  • 首先action的跳轉大致歸類: 1跳轉到與當前同一控制器內的action和不同控制器內的action、 2帶有參數的action跳轉和不帶參數的action跳轉。 3跳轉到指定視圖,不經過Controller的Action。 //跳轉到當前Controller的指定Action(此處為Index) ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...