1.概述

在面向对象程序设计过程中,程序员常常会遇到这种情况:设计一个系统时知道了算法所需的关键步骤,而且确定了这些步骤的执行顺序,但某些步骤的具体实现还未知,或者说某些步骤的实现与具体的环境相关。

例如,去银行办理业务一般要经过以下4个流程:取号、排队、办理具体业务、对银行工作人员进行评分等,其中取号、排队和对银行工作人员进行评分的业务对每个客户是一样的,可以在父类中实现,但是办理具体业务却因人而异,它可能是存款、取款或者转账等,可以延迟到子类中实现。

定义:定义一个操作中的算法骨架,而将算法的一些步骤延迟到子类中,使得子类可以不改变该算法结构的情况下重定义该算法的某些特定步骤。

2.结构

模板方法(Template Method)模式包含以下主要角色:

  • 抽象类(Abstract Class):负责给出一个算法的轮廓和骨架。它由一个模板方法和若干个基本方法构成。
    (1)模板方法:定义了算法的骨架,按某种顺序调用其包含的基本方法。
    (2)基本方法:是实现算法各个步骤的方法,是模板方法的组成部分。基本方法又可以分为三种:

    • 抽象方法(Abstract Method) :一个抽象方法由抽象类声明、由其具体子类实现
    • 具体方法(Concrete Method) :一个具体方法由一个抽象类或具体类声明并实现,其子类可以进行覆盖也可以直接继承。
    • 钩子方法(Hook Method) :在抽象类中已经实现,包括用于判断的逻辑方法和需要子类重写的空方法两种。一般钩子方法是用于判断的逻辑方法,这类方法名一般为isXxx,返回值类型为boolean类型。
  • 具体子类(Concrete Class):实现抽象类中所定义的抽象方法和钩子方法,它们是一个顶级逻辑的组成步骤。

3.案例实现

【例】炒菜

炒菜的步骤是固定的,分为倒油、热油、倒蔬菜、倒调料品、翻炒等步骤。现通过模板方法模式来用代码模拟。类图如下:

代码如下:

package com.itheima.pattern.template;/*** @program: design-patterns* @ClassName AbstractClass* @description: 抽象类:定义模板方法和基本方法* @author: * @create: 2023-01-24 17:03* @Version 1.0**/
public abstract class AbstractClass {// 模板方法定义public final void cookProcess() {pourOil();heatOil();pourVegetable();pourSauce();fry();}// 倒油public void pourOil() {System.out.println("倒油");}// 热油public void heatOil() {System.out.println("热油");}// 倒蔬菜public abstract void pourVegetable();// 倒调料public abstract void pourSauce();// 翻炒public void fry() {System.out.println("炒菜");}
}
package com.itheima.pattern.template;/*** @program: design-patterns* @ClassName ConcreteClass_BaoCai* @description: 爆炒包菜* @author: * @create: 2023-01-24 17:10* @Version 1.0**/
public class ConcreteClass_BaoCai extends AbstractClass {@Overridepublic void pourVegetable() {System.out.println("下锅的蔬菜是包菜");}@Overridepublic void pourSauce() {System.out.println("下锅的调料是辣椒");}
}
package com.itheima.pattern.template;/*** @program: design-patterns* @ClassName ConcreteClass_BaoCai* @description: 爆炒菜心* @author: * @create: 2023-01-24 17:10* @Version 1.0**/
public class ConcreteClass_CaiXin extends AbstractClass {@Overridepublic void pourVegetable() {System.out.println("下锅的蔬菜是菜心");}@Overridepublic void pourSauce() {System.out.println("下锅的调料是蒜蓉");}
}
package com.itheima.pattern.template;/*** @program: design-patterns* @ClassName Client* @description: 测试类* @author: * @create: 2023-01-24 17:12* @Version 1.0**/
public class Client {public static void main(String[] args) {// 创建炒菜的对象ConcreteClass_BaoCai baoCai = new ConcreteClass_BaoCai();// 调用炒菜的方法baoCai.cookProcess();}
}
注意:为防止恶意操作,一般模板方法都加上 final 关键词。

3.优缺点

  • 优点

    • 提高代码复用性,将相同部分的代码放在抽象的父类中,而将不同的代码放入不同的子类中。
    • 实现了反向控制,通过一个父类调用其子类的操作,通过对子类的具体实现扩展不同的行为,实现了反向控制 ,并符合“开闭原则”。
  • 缺点
    • 对每个不同的实现都需要定义一个子类,这会导致类的个数增加,系统更加庞大,设计也更加抽象。
    • 父类中的抽象方法由子类实现,子类执行的结果会影响父类的结果,这导致一种反向的控制结构,它提高了代码阅读的难度。

4.适用场景

  • 算法的整体步骤很固定,但其中个别部分易变时,这时候可以使用模板方法模式,将容易变的部分抽象出来,供子类实现。
  • 需要通过子类来决定父类算法中某个步骤是否执行,实现子类对父类的反向控制。

5.JDK源码解析

InputStream类就使用了模板方法模式。在InputStream类中定义了多个 read() 方法,如下:

public abstract class InputStream implements Closeable {//抽象方法,要求子类必须重写public abstract int read() throws IOException;public int read(byte b[]) throws IOException {return read(b, 0, b.length);}public int read(byte b[], int off, int len) throws IOException {if (b == null) {throw new NullPointerException();} else if (off < 0 || len < 0 || len > b.length - off) {throw new IndexOutOfBoundsException();} else if (len == 0) {return 0;}int c = read(); //调用了无参的read方法,该方法是每次读取一个字节数据if (c == -1) {return -1;}b[off] = (byte)c;int i = 1;try {for (; i < len ; i++) {c = read();if (c == -1) {break;}b[off + i] = (byte)c;}} catch (IOException ee) {}return i;}
}

从上面代码可以看到,无参的 read() 方法是抽象方法,要求子类必须实现。而 read(byte b[])方法调用了 read(byte b[], int off, int len) 方法,所以在此处重点看的方法是带三个参数的方法。

在该方法中第18行、27行,可以看到调用了无参的抽象的 read() 方法。

总结如下: 在InputStream父类中已经定义好了读取一个字节数组数据的方法是每次读取一个字节,并将其存储到数组的第一个索引位置,读取len个字节数据。具体如何读取一个字节数据呢?由子类实现。

行为型模式-模板方法模式相关推荐

  1. Java设计模式之行为型:模板方法模式

    一.什么是模板方法模式: 模板方法是基于继承实现的,在抽象父类中声明一个模板方法,并在模板方法中定义算法的执行步骤(即算法骨架).在模板方法模式中,可以将子类共性的部分放在父类中实现,而特性的部分延迟 ...

  2. 行为型设计模式---模板方法模式

    设计模式 序号 内容 链接地址 1 设计模式七大原则 https://blog.csdn.net/qq_39668819/article/details/115390615 2 创建型设计模式–工厂模 ...

  3. 设计模式 — 行为型模式 — 模板方法模式

    目录 文章目录 目录 模板方法模式 应用场景 代码示例 模板方法模式 模板方法模式,定义一个算法或者流程,部分环节设计为外部可变,用类似于模板的思想来实例化一个实体,可以往模板中填充不同的内容: 在模 ...

  4. 《设计模式详解》行为型模式 - 模板方法模式

    模板方法模式 6.1 模板方法模式 6.1.1 概述 6.1.2 结构 6.1.3 案例实现 6.1.3 优缺点 6.1.4 适用场景 6.1.5 JDK源码 - InputStream 完整的笔记目 ...

  5. 【设计模式】行为型02模板方法模式(Template Method Patten)

    五一长假,没有出去,不喜欢嘈杂的人群,玩了会游戏发泄了下憋在心底的戾气,手旁大马克杯里是母亲泡的绿茶.点开自己的播放列表,耳机里传来的是理查德克莱德曼的致爱丽丝.自己是个凡人,卑微渺小的活着.不说废话 ...

  6. Java设计模式(代理模式-模板方法模式-命令模式)

    Java设计模式Ⅴ 1.代理模式 1.1 代理模式概述 1.2 静态代理 1.2.1 静态代理概述 1.2.2 代码理解 1.3 动态代理之JDK代理 1.3.1 动态代理之JDK代理概述 1.3.2 ...

  7. 聊聊那些专为算法设计的模式——模板方法模式

    AI越来越火热,人工智能已然成风!而人工智能最重要是各种算法,因此机器学习越来越受到追捧,算法越来越被重视. 作为一个算法的研究者,写出一手高级算法当然是令人兴奋的一件事!但你是否有时会有这种感觉: ...

  8. 李建忠设计模式-组件协作模式-模板方法模式

    目录 1.前言 2.模板方法模式(Template Method) 1.动机 2.例子 3.定义 1.结构图 2.模板方法模式适用情形 3.模式特点 参考 1.前言 现代软件专业分工后的第一个结果是& ...

  9. java设计模式-- 观察者模式 , 外观模式, 模板方法模式

    观察者模式: 观察者模式比较容易理解: 比如天龙八部里面的王语嫣,每次慕容复打架的时候,每当别人将要使出什么绝招时,会提醒慕容复注意提防. 项目中用到的地方就是注册中中心,zookeeper,当Pro ...

最新文章

  1. 高质量程序程序设计指南摘录
  2. Winform DataGridView列的单元格中动态添加图片和文字
  3. (转载)机器学习知识点(二十九)LDA入门级学习笔记
  4. Python Tensorflow神经网络实现股票预测
  5. 用ESP32玩转真彩屏
  6. python 发布啦!!
  7. MySQL授权命令grant的使用方法
  8. 测试点3的分析:1022 D进制的A+B (20分)——15行代码AC
  9. python字符串类库_Python开发以太坊的类库Web3.py V4的新功能
  10. 问题:AmqpConnectException: java.net.ConnectException: Connection refused: connect
  11. 2018牛客多校第一场 B.Symmetric Matrix
  12. python写一个crm系统_用Python打造一个CRM系统(四)
  13. 地平线获近亿美元A+轮融资,AI时代的英特尔被英特尔领投
  14. 文件摆渡是什么意思_小白学炒股:除权、填权、贴权是个什么东东?
  15. iptables和SELinux漫谈
  16. 简单提高MIDI音量的方法
  17. java转盘抽奖算法_抽奖算法 - 幸运大转盘 - java
  18. 如何高效读懂一本书(附10种读书方法)
  19. ReactJs 简介
  20. Matlab syms 矩阵变量,matlab syms.m

热门文章

  1. CAD图纸导入到PS的步骤
  2. c语言标志位设置原理
  3. 网卡引起的网络浏览速度变慢
  4. 轮廓波-非下采样轮廓波NSCT
  5. 程序员炒股,如何计算股票投资组合的风险和收益
  6. 基于python的问答对联生成系统 附完整代码 毕业设计
  7. Typora修改字体颜色
  8. 深入浅出,camera v4l2理解
  9. 为啥离不了 linux
  10. 基于sp++ matlab hanning窗 C++ 实现