模板指一些可以套用的公共内容,例如网页模板是当网站中有许多页面版式色彩相同的情况下,将其定义为网页模板,并定义其中部分可编辑,部分不可编辑,那么在利用网页模板制作其他页面时就会很方便,不易出错。
在设计模式中,模板方法模式中模板和生活中模板概念非常类似,在一个抽象类中定义一个操作中的算法骨架(对应于模板),而将一些步骤延迟到子类中去实现(对应根据自己的情况向模板填充内容)。
在面向对象程序设计过程中,程序员常常会遇到这种情况:设计一个系统时知道了算法所需的关键步骤,而且确定了这些步骤的执行顺序,但某些步骤的具体实现还未知,或者说某些步骤的实现与具体的环境相关。此时就可以采用模板方法进行设计。
例如公司的入职流程,进入公司,入职准备、入职报到、办理入职手续、进行入职培训,转正,入职结束进入岗位。这些步骤都很固定,但是不同的公司,流程中每个步骤稍有不同。这些不同的可以在具体实现上进行填充。
特点优点:
它封装了不变部分,扩展可变部分。它把认为是不变部分的算法封装到父类中实现,而把可变部分算法由子类继承实现,便于子类继续扩展。
它在父类中提取了公共的部分代码,实现了代码复用。
部分方法是由子类实现的,因此子类可以通过扩展方式增加相应的功能,符合开闭原则。
缺点:
对每个不同的实现都需要定义一个子类,这会导致类的个数增加,系统更加庞大,设计也更加抽象。
父类中的抽象方法由子类实现,子类执行的结果会影响父类的结果,这导致一种反向的控制结构,它提高了代码阅读的难度。
因为引入了一个抽象类,如果具体实现过多的话,需要用户或开发人员需要花更多的时间去理清类之间的关系。
根据模板方法模式类图结构,有利于我们理清该模式中类之间的关系,具体类图如下:
模板方法模式中涉及的角色:
抽象模板角色:定义了一个或多个抽象操作,以便让子类实现,这些抽象操作称为基本操作。它由一个模板方法和若干个基本方法构成。
模板方法:定义了算法的骨架,按某种顺序调用其包含的基本方法。
基本方法:是整个算法中的一个步骤,包含了抽象方法、具体方法。
具体模板角色:实现父类所定义的一个或多个抽象方法。
算法的整体步骤很固定,但其中个别部分易变时,这时候可以使用模板方法模式,将容易变的部分抽象出来,供子类实现。
当多个子类存在公共的行为时,可以将其提取出来并集中到一个公共父类中以避免代码重复。首先,要识别现有代码中的不同之处,并且将不同之处分离为新的操作。最后,用一个调用这些新的操作的模板方法来替换这些不同的代码。
以入职流程为例,进行具体的实现。
using System; namespace 模板模式 { class EntryProcess { static void Main(string[] args) { SamsungEntryProcess samsung = new SamsungEntryProcess(); HuaweiEntryProcess huawei = new HuaweiEntryProcess(); samsung.JoiningCompany(); huawei.JoiningCompany(); Console.Read(); } } // 抽象类,入职流程 public abstract class TemplateEntryProcess { // 模板方法,不要把模版方法定义为 Virtual 或 abstract 方法,避免被子类重写,防止更改流程的执行顺序 public void JoiningCompany() { this.entryCompany(); // 进入公司 this.preparationForEntry(); // 入职前准备,整理衣帽等 this.registrationForEmployment(); // 入职报到 this.entryProcedures(); // 办理入职手续 this.inductionTraining(); // 入职培训 this.evaluationOfConversion(); // 转正评估 this.entryOver(); // 入职结束,进入岗位 } public abstract void entryCompany(); public void preparationForEntry() { Console.WriteLine("做准备,整理衣帽等;"); } public void registrationForEmployment() { Console.WriteLine("入职报到;"); } public void entryProcedures() { Console.WriteLine("办理入职手续;"); } public void inductionTraining() { Console.WriteLine("进行入职培训;"); } public void evaluationOfConversion() { Console.WriteLine("完成入职前培训,进行转正评估;"); } public void entryOver() { Console.WriteLine("入职流程完成,进入岗位。"); } } // 具体子类,三星入职 public class SamsungEntryProcess: TemplateEntryProcess { public override void entryCompany() { Console.WriteLine("进入三星公司"); } } // 具体子类,华为入职 public class HuaweiEntryProcess: TemplateEntryProcess { public override void entryCompany() { Console.WriteLine("进入华为公司"); } } }