前言

模板方法模式,定义一个操作中算法的骨架,而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重新定义该算法的某些特定步骤。

模板方法模式

概念介绍

模板方法模式,其实是很好理解的,具体理解为,定义一个操作中算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法结构即可冲定义该算法的某些特定步骤。模板方法模式是最为常见的设计模式之一,是基于继承的代码复用技术。架构师定义一套骨架,开发工程师按照骨架去实现具体的逻辑。

举例

在具体的项目中其实使用模板方法的场景有很多,就举我之间遇到过的一个场景吧,这是一个实际的项目,结合这个项目的中的场景可以引出模板方法模式的使用方法,

在一个审批流程的项目,会有很多种的申请单,例如:出差申请单、请假申请单、采购申请单、付款申请单等等。每个申请单在提交的时候都要先校验申请单的参数是否正确,以及在提交成功后去通知审批人,有待审批的单子了。所以说,这个过程都是一样的,只不过是,每个申请单的校验参数以及通知审批人的方法会有所不同。那么这样就可以把这个提交的过程定义成一个模板,然后每个审批单的都按照这个流程来进行提交申请就可以了。

具体代码实现如下:

流程模板类

/**
* 流程模板类
*/
public abstract class BaseProcess { /**
* 提交流程
*/
public void submitProcess(Map<String,String> paramMap){
boolean checkResult = checkParameter(paramMap);
if(checkResult){
System.out.println("提交流程成功!");
remindApprovers();
}else {
System.out.println("提交流程失败!");
}
} /**
* 校验参数
* @return
*/
public abstract boolean checkParameter(Map<String,String> paramMap); /**
* 提醒审批人
*/
public abstract void remindApprovers();
}

差旅审批流程

/**
* 差旅审批流程
*
*/
public class BusinessTravelProcess extends BaseProcess{
/**
* 校验参数
*
* @return
*/
@Override
public boolean checkParameter(Map<String,String> paramMap) { if(null!=paramMap.get("result")&&paramMap.get("result").equals("true")){
System.out.println("差旅审批单参数校验成功!");
return true;
}else {
System.out.println("差旅审批单参数校验失败!");
return false;
}
} /**
* 提醒审批人
*/
@Override
public void remindApprovers() {
System.out.println("有新的差旅申请提交了。");
}
}

请假审批流程

/**
* 请假审批流程
*/
public class LeaveApplyProcess extends BaseProcess {
/**
* 校验参数
*
* @return
*/
@Override
public boolean checkParameter(Map<String,String> paramMap) { if(null!=paramMap.get("result")&&paramMap.get("result").equals("true")){
System.out.println("请假审批单参数校验成功!");
return true;
}else {
System.out.println("请假审批单参数校验成功!");
return false;
} } /**
* 提醒审批人
*/
@Override
public void remindApprovers() {
System.out.println("有新的请假申请提交了。");
}
}

测试类

public class Client {

    public static void main(String[] args) {

        //创建请假申请单
BaseProcess leaveApply = new LeaveApplyProcess();
Map<String,String> paramMap = Maps.newHashMap();
paramMap.put("result","true");
//提交采购申请单
leaveApply.submitProcess(paramMap);
//创建差旅申请单
BaseProcess business = new BusinessTravelProcess();
paramMap = Maps.newHashMap();
paramMap.put("result","false");
//提交差旅申请单
business.submitProcess(paramMap); } }

运行结果

请假审批单参数校验成功!
提交流程成功!
有新的请假申请提交了。
差旅审批单参数校验失败!
提交流程失败!

上面的这个例子就是使用的模板方法模式,这个场景是一些业务功能,大体框架是固定的,只是一些具体的实现细节可能不同。用模板方法能提高代码的复用性和系统的灵活性。

结构分析

下面来分析一下模板方法的具体结构组成,如下是模板方法模式的类图。

模板方法模式中,具体就两个角色。

1、AbstractClass(抽象类):在抽象类中定义了一系列基本操作(PrimitiveOperations),这些基本操作可以是具体的,也可以是抽象的,每一个基本操作对应算法的一个步骤,在其子类中可以重定义或实现这些步骤。

2、ConcreteClass(具体子类):它是抽象类的子类,用于实现在父类中声明的抽象基本操作以完成子类特定算法的步骤,也可以覆盖在父类中已经实现的具体基本操作。

但是模板方法模式的实现是离不开这三个方法的

1、基本方法

基本方法也称为基本操作,是由子类实现的方法,并且在模板方法中被调用。

2、模板方法

模板方法可以有一个或几个,一般是一个具体的方法,也就是一个骨架,实现对基本方法的调度,完成固定的逻辑。为了防止恶意的操作,一般模板方法都加上final关键字,不允许被覆写。

3、钩子方法

钩子方法由抽象类声明并加以实现。但是子类可以去扩展,子类可以通过扩展钩子方法,来影响模板方法的逻辑。抽象类的任务是搭建逻辑的框架,通常由经验丰富的人员编写,因为抽象类的好坏直接决定了程序是否稳定。

总结

模板方法模式是基于继承的代码复用技术,它体现了面向对象的诸多重要思想,是一种使用较为频繁的模式。模板方法模式广泛应用于框架设计中,以确保通过父类来控制处理流程的逻辑顺序(如框架的初始化,测试流程的设置等)。

主要优点

1、在父类中形式化地定义一个算法,而由它的子类来实现细节的处理,在子类实现详细的处理算法时并不会改变算法中步骤的执行次序。

2、模板方法模式是一种代码复用技术,它在类库设计中尤为重要,它提取了类库中的公共行为,将公共行为放在父类中,而通过其子类来实现不同的行为,它鼓励我们恰当使用继承来实现代码复用。

3、可实现一种反向控制结构,通过子类覆盖父类的钩子方法来决定某一特定步骤是否需要执行。

4、在模板方法模式中可以通过子类来覆盖父类的基本方法,不同的子类可以提供基本方法的不同实现,更换和增加新的子类很方便,符合单一职责原则和开闭原则。

主要缺点

1、需要为每一个基本方法的不同实现提供一个子类,如果父类中可变的基本方法太多,将会导致类的个数增加,系统更加庞大,设计也更加抽象,此时,也可结合桥接模式来进行设计。

2、由于每个子类的方法会影响到了父类,这里违反了里氏替换原则,会给程序带来风险。

适用场景

1、对一些复杂的算法进行分割,将其算法中固定不变的部分设计为模板方法和父类具体方法,而一些可以改变的细节由其子类来实现。

2、各子类中公共的行为应被提取出来并集中到一个公共父类中以避免代码重复。

3、需要通过子类来决定父类算法中某个步骤是否执行,实现子类对父类的反向控制。

想了解更多的设计模式请查看Java设计模式学习记录-GoF设计模式概述

Java设计模式学习记录-模板方法模式的更多相关文章

  1. Java设计模式学习记录-状态模式

    前言 状态模式是一种行为模式,用于解决系统中复杂的对象状态转换以及各个状态下的封装等问题.状态模式是将一个对象的状态从该对象中分离出来,封装到专门的状态类中,使得对象的状态可以灵活多变.这样在客户端使 ...

  2. Java设计模式学习记录-备忘录模式

    前言 这次要介绍的是备忘录模式,也是行为模式的一种 .现在人们的智能手机上都会有备忘录这样一个功能,大家也都会用,就是为了记住某件事情,防止以后自己忘记了.那么备忘录模式又是什么样子的呢?是不是和手机 ...

  3. Java设计模式学习记录-迭代器模式

    前言 这次要介绍的是迭代器模式,也是一种行为模式.我现在觉得写博客有点应付了,前阵子一天一篇,感觉这样其实有点没理解透彻就写下来了,而且写完后自己也没有多看几遍,上次在面试的时候被问到java中的I/ ...

  4. Java设计模式学习记录-解释器模式

    前言 这次介绍另一个行为模式,解释器模式,都说解释器模式用的少,其实只是我们在日常的开发中用的少,但是一些开源框架中还是能见到它的影子,例如:spring的spEL表达式在解析时就用到了解释器模式,以 ...

  5. Java设计模式学习记录-命令模式

    前言 这次要介绍的是命令模式,这也是一种行为型模式.最近反正没有面试机会我就写博客呗,该投的简历都投了.然后就继续看书,其实看书也会给自己带来成就感,原来以前不明白的东西,书上已经给彻底的介绍清楚了, ...

  6. Java设计模式学习记录-外观模式

    前言 这次要介绍的是外观模式(也称为门面模式),外观模式也属于结构型模式,其实外观模式还是非常好理解的,简单的来讲就是将多个复杂的业务封装成一个方法,在调用此方法时可以不必关系具体执行了哪些业务,而只 ...

  7. Java设计模式学习记录-桥接模式

    前言 这次介绍结构型设计模式中的第二种模式,桥接模式. 使用桥接模式的目的就是为了解耦,松散的耦合更利于扩展,但是会增加相应的代码量和设计难度. 桥接模式 桥接模式是为了将抽象化与实现化解耦,让二者可 ...

  8. Java设计模式学习记录-代理模式

    代理模式 代理模式是常见设计模式的一种,代理模式的定义是:为其他对象提供一种代理以控制对这个对象的访问. 在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起 ...

  9. Java设计模式学习记录-建造者模式

    前言 今天周末,有小雨,正好也不用出门了,那就在家学习吧,经过了两周的面试,拿到了几个offer,但是都不是自己很想去的那种,要么就是几个人的初创小公司,要么就是开发企业内部系统的这种传统开发,感觉这 ...

随机推荐

  1. 压力测试工具 Apache_jmeter软件配置+TCP示例说明

    该软件jmeter是Apache官方开源压力测试软件.  jmeter官网:http://jmeter.apache.org/  . 本文使用的版本是 3.0版本, 它需要jdk7及以上版本支持. 网 ...

  2. shell awk处理过滤100万条数据

    背景: 100万条数据.格式如下: ID 地址 1895756546931805 安徽省六安市裕安区固镇镇佛俺村柳树队5758 安徽省蒙城县岳坊镇胡寨村小组小胡寨庄6号 183494167409969 ...

  3. HTML常用的特殊符号&前端使用的标点符号

    不想在项目中使用图片, 还得切,如关闭按钮“×”.男女符号“♂♀”.对勾“√”等,找到了一篇全面的博客,转自https://www.haorooms.com/post/html_tsfh,感谢. 如下 ...

  4. 2019浙大校赛--E--Potion(签到水题)

    一丢丢思维就ok 题目大意: 魔法师要煮药,有n个等级的药,所需要的药物为a1,a2...an,意为第n级需要多少药物,下一行为库存的不同等级药物,药物可降级使用不可升级. 思路:从高级药物开始解,把 ...

  5. Java File类与文件IO流总结

    1.File类 File类被定义为“文件和目录路径名的抽象表示形式”,这是因为File类既可以表示“文件”也可以表示“目录”,他们都通过对应的路径来描述.通过构造函数创建一个File类对象,则该对象就 ...

  6. 开机后Android应用自动启动

    一.需求 在应用开发过程中,有客户提出在设备开机后自动启动应用. 二.实现方法 实现方案:安卓系统每次开机的时候都会发送一个广播,监听这个广播,广播事件触发启动应用程序. 监听音频广播而不是启动广播, ...

  7. grep 笔记

    -a :将 binary 文件以 text 文件的方式搜寻数据-c :计算找到 '搜寻字符串' 的次数-i :忽略大小写的不同,所以大小写视为相同-n :顺便输出行号-v :反向选择,亦即显示出没有 ...

  8. linux之配置Mongodb~

       OK 让我们先下载一波mongodb~(64位ubuntu) curl -O https://fastdl.mongodb.org/linux/mongodb-linux-x86_64-3.0. ...

  9. Senparc.Weixin.TenPay 正式发布

    微信支付刚出来的时候,和公众号的绑定关系很深(甚至旧版本使用的就是公众号的appId),随着微信生态的逐步丰富,微信支付越来越成为一个独立的平台,同时服务于公众号.小程序.开放平台.企业号/企业微信等 ...

  10. MapReduce多种join实现实例分析(二)

    上一篇<MapReduce多种join实现实例分析(一)>,大家可以点击回顾该篇文章.本文是MapReduce系列第二篇. 一.在Map端进行连接使用场景:一张表十分小.一张表很大.用法: ...