设计模式-模板方法
一、?模板方法(Template Method)模式
準備一個抽象類,將部分邏輯以具體方法以及具體構造子的形式實現,然后聲明一些抽象方法來迫使子類實現剩余的邏輯。不同的子類可以以不同的方式實現這些抽象方法,從而對剩余的邏輯有不同的實現。這就是模版方法模式的用意。
很多人可能沒有想到,模版方法模式實際上是所有模式中最為常見的幾個模式之一,而且很多人可能使用過模版方法模式而沒有意識到自己已經使用了這個模式。模版方法模式是基于繼承的代碼復用的基本技術,模版方法模式的結構和用法也是面向對象設計的核心。
模 版方法模式需要開發抽象類和具體子類的設計師之間的協作。一個設計師負責給出一個算法的輪廓和骨架,另一些設計師則負責給出這個算法的各個邏輯步驟。代表 這些具體邏輯步驟的方法稱做基本方法(primitive method);而將這些基本法方法總匯起來的方法叫做模版方法(template method),這個設計模式的名字就是從此而來。
二、?模版方法模式的結構
模版方法模式的靜態結構如下圖所示。
?
這里涉及到兩個角色:
- 抽象模版(AbstractClass)角色有如下的責任:
定義了一個或多個抽象操作,以便讓子類實現。這些抽象操作叫做基本操作,它們是一個頂級邏輯的組成步驟。
定義并實現了一個模版方法。這個模版方法一般是一個具體方法,它給出了一個頂級邏輯的骨架,而邏輯的組成步驟在相應的抽象操作中,推遲到子類實現。頂級邏輯也有可能調用一些具體方法。
- 具體模版(ConcreteClass)角色有如下的責任:
實現父類所定義的一個或多個抽象方法,它們是一個頂級邏輯的組成步驟。
每一個抽象模版角色都可以有任意多個具體模版角色與之對應,而每一個具體模版角色都可以給出這些抽象方法(也就是頂級邏輯的組成步驟)的不同實現,從而使得頂級邏輯的實現各不相同。
三、?模板方法模式的示意性代碼
//?Template?Method?pattern?--?Structural?example??using?System;
//?"AbstractClass"
abstract?class?AbstractClass
{
??//?Methods
??abstract?public?void?PrimitiveOperation1();
??abstract?public?void?PrimitiveOperation2();
??//?The?Template?method
??public?void?TemplateMethod()
??{
????Console.WriteLine("In?AbstractClass.TemplateMethod()");
????PrimitiveOperation1();
????PrimitiveOperation2();
??}
}
//?"ConcreteClass"
class?ConcreteClass?:?AbstractClass
{
??//?Methods
??public?override?void?PrimitiveOperation1()
??{
????Console.WriteLine("Called?ConcreteClass.PrimitiveOperation1()");
??}
??public?override?void?PrimitiveOperation2()
??{
????Console.WriteLine("Called?ConcreteClass.PrimitiveOperation2()");
??}
}
///?<summary>
///?Client?test
///?</summary>
public?class?Client
{
??public?static?void?Main(?string[]?args?)
??{
????//?Create?instance?and?call?template?method
????ConcreteClass?c?=?new?ConcreteClass();
????c.TemplateMethod();
??}
}
四、?繼承作為復用的工具
使用繼承作為復用的手段必須慎重,C#語言的設計師對使用繼承作為復用的工具有著不同層次上的認識。
不知其一
首先,初學C#的程序員可能不知道什么是繼承,或者認為"繼承"是高深的工具。那時候,大部分的功能復用都是通過委派進行的。
知其一、不知其二
然后慢慢地,他們發現在C#語言里實現繼承并不困難,并且初步認識到繼承可以使子類一下子得到基類的行為。這時他們就會躍躍欲試了,試圖使用繼承作為功能復用的主要工具,并把原來應當使用委派的地方,改為使用繼承,這時繼承就有被濫用的危險。
知其二
很多面向對象的設計專家從1986年就開始警告繼承關系被濫用的可能。有一些面向對象的編程語言,如SELF語言,甚至將類的繼承關系從語言的功能中取消掉,改為完全使用委派。
其 他的設計師雖然不提倡徹底取消繼承,但無一例外地鼓勵在設計中盡可能使甩委派關系代替繼承關系。比如在【GOF95】一書中,狀態模式、策略模式、裝飾模 式、橋梁模式以及抽象工廠模式均是將依賴于繼承的實現轉換為基于對象的組合和聚合的實現,這些模式的要點就是使用委派關系代替繼承關系。
知其三
是不是繼承就根本不該使用呢?事實上對數據的抽象化、繼承、封裝和多態性并稱C#和其他絕大多數的面向對象語言的幾項最重要的特性。繼承不應當被濫用,并不意味著繼承根本就不該使用。因為繼承容易被濫用就徹底拋棄繼承,無異于因噎廢食。
繼 承使得類型的等級結構易于理解、維護和擴展,而類型的等級結構非常適合于抽象化的設計、實現和復用。盡管【GOF95】所給出的設計模式基本上沒有太多基 于繼承的模式,很多模式都是用繼承的辦法定義、實現接口的。多數的設計模式都描寫一個以抽象類作為基類,以具體類作為實現的等級結構,比如適配器模式、合 成模式、橋梁模式、狀態模式等。
模版方法模式則更進了一步:此模式鼓勵恰當地使用繼承。此模式可以用來改寫一些擁有相同功能的相關的類,將可復用的一般性的行為代碼移到基類里面,而把特殊化的行為代碼移到子類里面。
因此,熟悉模版方法模式便成為一個重新學習繼承的好地方。
五、?一個實際應用模板方法的例子
下面的例子演示了數據庫訪問的模板方法。實際應用時,請確保C盤根目錄下有nwind.mdb這個Access數據庫(可以從Office的安裝目錄下找到。中文版用戶的請注意字段名可能有所不同)。
//?Template?Method?pattern?--?Real?World?example??using?System;
using?System.Data;
using?System.Data.OleDb;
//?"AbstractClass"
abstract?class?DataObject
{
??//?Methods
??abstract?public?void?Connect();
??abstract?public?void?Select();
??abstract?public?void?Process();
??abstract?public?void?Disconnect();
??//?The?"Template?Method"
??public?void?Run()
??{
????Connect();
????Select();
????Process();
????Disconnect();
??}
}
//?"ConcreteClass"
class?CustomerDataObject?:?DataObject
{
??private?string?connectionString?=
????"provider=Microsoft.JET.OLEDB.4.0;?"
????+?"data?source=c:\\nwind.mdb";
??private?string?commandString;
??private?DataSet?dataSet;
?
??//?Methods
??public?override?void?Connect(?)
??{
????//?Nothing?to?do
??}
??public?override?void?Select(?)
??{
????commandString?=?"select?CompanyName?from?Customers";
????OleDbDataAdapter?dataAdapter?=?new?OleDbDataAdapter(
??????commandString,?connectionString?);
????dataSet?=?new?DataSet();
????dataAdapter.Fill(?dataSet,?"Customers"?);
??}
??public?override?void?Process()
??{
????DataTable?dataTable?=?dataSet.Tables["Customers"];
????foreach(?DataRow?dataRow?in?dataTable.Rows?)
??????Console.WriteLine(?dataRow[?"CompanyName"?]?);
??}
??public?override?void?Disconnect()
??{
????//?Nothing?to?do
??}
}
///?<summary>
///??TemplateMethodApp?test
///?</summary>
public?class?TemplateMethodApp
{
??public?static?void?Main(?string[]?args?)
??{
????CustomerDataObject?c?=?new?CustomerDataObject(?);
????c.Run();
??}
}
?
六、?模版方法模式中的方法
模版方法中的方法可以分為兩大類:模版方法(Template Method)和基本方法(Primitive Method)。
模版方法
一個模版方法是定義在抽象類中的,把基本操作方法組合在一起形成一個總算法或一個總行為的方法。這個模版方法一般會在抽象類中定義,并由子類不加以修改地完全繼承下來。
基本方法
基本方法又可以分為三種:抽象方法(Abstract Method)、具體方法(Concrete Method)和鉤子方法(Hook Method)。
抽象方法:一個抽象方法由抽象類聲明,由具體子類實現。在C#語言里一個抽象方法以abstract關鍵字標示出來。
具體方法:一個具體方法由抽象類聲明并實現,而子類并不實現或置換。在C#語言里面,一個具體方法沒有abstract關鍵字。
鉤 子方法:一個鉤子方法由抽象類聲明并實現,而子類會加以擴展。通常抽象類給出的實現是一個空實現,作為方法的默認實現。(Visual FoxPro中項目向導建立的項目會使用一個AppHook類實現監視項目成員變化,調整系統結構的工作。)鉤子方法的名字通常以do開始。
七、?重構的原則
在對一個繼承的等級結構做重構時,一個應當遵從的原則便是將行為盡量移動到結構的高端,而將狀態盡量移動到結構的低端。
1995年,Auer曾在文獻【AUER95】中指出:
如果能夠遵從這樣的原則,那么就可以在等級結構中將接口與實現分隔開來,將抽象與具體分割開來,從而保證代碼可以最大限度地被復用。這個過程實際上是將設計師引導到模版方法模式上去。
轉載于:https://www.cnblogs.com/zhuxiongfeng/archive/2010/04/09/1708615.html
總結
- 上一篇: lds语法规则
- 下一篇: 面向对象与面向过程在软件开发中的应用