设计模式-模版方法模式

一、定义

模版方法模式是一种对象行为模式。定义一个抽象类,将部分逻辑以具体方法(算法骨架)及具体构造函数的形式实现,然后声明一些抽象方法迫使子类实现剩余逻辑;不同子类可以以不同方式实现这些抽象方法,从而对剩余的逻辑有不同的实现。

二、优点

1、封装了不变部分,扩展了可变部分,符合开闭原则。
2、提取了公共代码,便于维护。
3、行为由父类控制,子类实现。

三、缺点

每一个不同的实现都需要定义一个子类,后期会导致类的个数增加,系统更加庞大。

四、模式结构与实现

1)模式的结构

抽象处理者(AbstractClass):定义算法骨架及部分由子类实现的抽象方法。
具体处理者(ConcreteClass):继承抽象处理者。具体实现抽象处理类中定义的抽象方法。
结构图:

模版方法模式结构图.png

2)模式的实现

//测试类
public class Test {
    public static void main(String[] args) {
        AbstractClass demo=new ConcreteClass1();
        demo.templateMethod();
        demo=new ConcreteClass2();
        demo.templateMethod();
    }
}
//抽象处理者角色
public abstract class AbstractClass {
    protected void templateMethod() {
        operation1();
        operation2();
    }
    abstract void operation1();
    abstract void operation2();
}
//具体处理者角色1
public class ConcreteClass1 extends AbstractClass {
    @Override
    public void operation1() {
        System.out.println("ConcreteClass1执行operation1");
    }
    @Override
    public void operation2() {
        System.out.println("ConcreteClass1执行operation2");
    }
}
//具体处理者角色2
public class ConcreteClass2 extends AbstractClass {
    @Override
    public void operation1() {
        System.out.println("ConcreteClass2执行operation1");
    }
    @Override
    public void operation2() {
        System.out.println("ConcreteClass2执行operation2");
    }
}

运行结果:

ConcreteClass1执行operation1
ConcreteClass1执行operation2
ConcreteClass2执行operation1
ConcreteClass2执行operation2

五、应用场景

1、多个子类共有的方法且处理逻辑基本相同。
2、重要、复杂的算法,可以把核心算法设计为模板方法,周边的相关细节功能则由各个子类实现。
3、重构时,模板方法是一个经常使用的方法,把相同的代码抽取到父类中,然后通过构造函数约束其行为。

六、应用实例

1)需求

用模版方法模式实现泡茶流程和煮咖啡流程

2)分析

泡咖啡流程
1、把水煮沸
2、用沸水冲泡咖啡
3、把咖啡倒进杯子
4、加糖和牛奶

泡茶流程
1、把水煮沸
2、用沸水冲泡茶叶
3、把茶倒进杯子
4、加柠檬

3)设计思路

1、抽象类
定义一个抽象类(CaffeineBeverage),它定义了泡茶(咖啡)的大致流程;实现了将水烧开,将水倒入杯中等部分流程;
2、子类
定义茶类(Tea)、咖啡类(Coffee),它们各自实现了用开水冲泡茶叶(咖啡);加入调味品等流程。
3、结构图

结构图.jpg

4)程序实现

//测试类
public class Test {
    public static void main(String[] args) {
        CaffeineBeverage demo=new Tea();
        demo.prepareRecipe();
        System.out.println("---------------------");
        demo=new Coffee();
        demo.prepareRecipe();
    }
}
//抽象处理者
public abstract class CaffeineBeverage {
    void prepareRecipe() {
        boilWater();
        brew();
        pourInCup();
        addCondiments();
    }
    abstract void addCondiments();
    abstract void brew();
    public void boilWater() {
        System.out.println("Boiling water");
    }
    public void pourInCup() {
        System.out.println("Pouring into cup");
    }
}
//具体处理者:茶
public class Tea extends CaffeineBeverage {
    @Override
    public void brew() {
        System.out.println("Steeping the tea");
    }
    @Override
    public void addCondiments() {
        System.out.println("Adding Lemon");
    }
}//具体处理者:咖啡
public class Coffee extends CaffeineBeverage {
    @Override
    public void addCondiments() {
        System.out.println("Dripping coffee through filter");
    }
    @Override
    public void brew() {
        System.out.println("Adding Sugar and Milk");
    }
}

程序运行结果如下:

Boiling water
Steeping the tea
Pouring into cup
Adding Lemon
---------------------
Boiling water
Adding Sugar and Milk
Pouring into cup
Dripping coffee through filter

七、模版方法模式在开源项目中的应用

1)Spring MVC
在SringMVC框架里,体现在HttpServletBeanFrameworkServletResourceServlet这3个类中。这三个类的关系如下:

模版方法模式应用.png

其中HttpServletBean是一个抽象类,继承了HttpServlet类,覆写了init方法,init方法实现里面又定义了一个initServletBean方法,但是未做任何实现。两个子类FrameworkServletResourceServlet分别以不同方式实现了该方法。
1、以下是HttpServletBean定义的init方法:

@Override
public final void init() throws ServletException {
    if (logger.isDebugEnabled()) {
        logger.debug("Initializing servlet '" + getServletName() + "'");
    }
    // Set bean properties from init parameters.
    try {
        PropertyValues pvs = new ServletConfigPropertyValues(getServletConfig(), this.requiredProperties);
        BeanWrapper bw = PropertyAccessorFactory.forBeanPropertyAccess(this);
        ResourceLoader resourceLoader = new ServletContextResourceLoader(getServletContext());
        bw.registerCustomEditor(Resource.class, new ResourceEditor(resourceLoader, getEnvironment()));
        initBeanWrapper(bw);
        bw.setPropertyValues(pvs, true);
    }
    catch (BeansException ex) {
        logger.error("Failed to set bean properties on servlet '" + getServletName() + "'", ex);
        throw ex;
    }
    // Let subclasses do whatever initialization they like.
    initServletBean();
    if (logger.isDebugEnabled()) {
        logger.debug("Servlet '" + getServletName() + "' configured successfully");
    }
}

initServletBean方法定义如下:

protected void initBeanWrapper(BeanWrapper bw) throws BeansException {}

2、子类FrameworkServlet实现initServletBean方法如下:

@Override
protected final void initServletBean() throws ServletException {
    getServletContext().log("Initializing Spring FrameworkServlet '" + getServletName() + "'");
    if (this.logger.isInfoEnabled()) {
        this.logger.info("FrameworkServlet '" + getServletName() + "': initialization started");
    }
    long startTime = System.currentTimeMillis();
    try {
        this.webApplicationContext = initWebApplicationContext();
        initFrameworkServlet();
    }
    catch (ServletException ex) {
        this.logger.error("Context initialization failed", ex);
        throw ex;
    }
    catch (RuntimeException ex) {
        this.logger.error("Context initialization failed", ex);
        throw ex;
    }
    if (this.logger.isInfoEnabled()) {
        long elapsedTime = System.currentTimeMillis() - startTime;
        this.logger.info("FrameworkServlet '" + getServletName() + "': initialization completed in " +
                elapsedTime + " ms");
    }
}

3、子类ResourceServlet实现initServletBean方法如下:

@Override
protected void initServletBean() {
    this.pathMatcher = getPathMatcher();
    this.startupTime = System.currentTimeMillis();
}

2)Spring集成Hibernate对数据库的操作流程
Spring中对 Hibernate 的支持,将一些已经定好的方法封装起来,比如开启事务、获取 Session、关闭 Session 等,程序员不重复写那些已经规范好的代码,直接丢一个实体就可以保存。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容