模板方法模式定义
模板方法模式定义:Define the skeleton of an algorithm in an operation, deferring some steps to subclasses. Template Method lets subclassed redefine certain steps of an algorithm without changing the algorithm's structure.(定义一个操作中的算法的框架,而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重新定义该算法的某写特定步骤。)
AbstractClass叫做抽象模版,它的方法分为两类,基本方法:基本方法也叫基本操作,是由子类实现的方法,并且在模版方法中被调用。模版方法:模版方法可以有一个或几个,一般是一个具体方法,也就是一个框架,实现对基本方法的调度,完成固定的逻辑。为了防止恶意操作,一般模版方法都加上final关键字,不允许被覆写。
模版方法模式应用场景
1、多个子类有共有的方法,并且逻辑基本相同时。
2、复杂、重要的算法,可以把核心算法设计为模版方法,周边的相关细节功能则油各个子类实现。
3、重构时,把相同的代码抽取到父类中,然后通过钩子函数约束其行为。
模版方法模式的优点:封装不变部分,扩展可变部分;提取公共代码,便于维护;行为由父类控制,子类实现。
模版方法模式的缺点:子类执行的结果会影响父类的结果,也就是子类会对父类产生影响。
体验模板方法模式
/** * 豆浆类 */ public abstract class SoyaMilk { //模板方法:选材方法,选择黄豆 void selectMaterial(){ System.out.println("第一步、选择新鲜黄豆"); } //基本方法:添加不同的配料,在这里设置为抽象方法,子类必须实现 abstract void addCondiments(); //模板方法:浸泡 void soak(){ System.out.println("第三步、浸泡"); } //基本方法:搅拌 void beat(){ System.out.println("第四步、黄豆的配料放到豆浆机打碎"); } //钩子方法:是否需要加原料 boolean hook(){ return true; } // 模板方法:用final修饰,不允许子类覆盖。模板方法定义了制作豆浆的程序 final void prepareRecipe(){ this.selectMaterial(); if (this.hook()){ this.addCondiments(); } this.soak(); this.beat(); } }
/** * 红枣豆浆 */ public class ReddatesSoyaMilk extends SoyaMilk { @Override void addCondiments() { System.out.println("第二步:添加红枣配料"); } }
/** * 核桃豆浆 */ public class NutSoyaMilk extends SoyaMilk { @Override void addCondiments() { System.out.println("第二步:添加核桃配料"); } }
/** * 原味豆浆 */ public class OriginalSoyaMilk extends SoyaMilk { @Override void addCondiments() { System.out.println("不加配料,我喜欢原味"); } @Override boolean hook(){ return false; } }
public class MainTest { @Test public void test(){ // 红枣豆浆 SoyaMilk reddatesSoyaMilk = new ReddatesSoyaMilk(); reddatesSoyaMilk.prepareRecipe(); // 核桃豆浆 SoyaMilk nutSoyaMilk = new NutSoyaMilk(); nutSoyaMilk.prepareRecipe(); // 原味豆浆 SoyaMilk originalSoyaMilk = new OriginalSoyaMilk(); originalSoyaMilk.prepareRecipe(); } }