原创作品,可以转载,但是请标注出处地址:https://www.cnblogs.com/V1haoge/p/9558825.html

  模板模式,顾名思义,就是通过模板拓印的方式。

  定义模板,就是定义框架、结构、原型。定义一个我们共同遵守的约定。

  定义了模板,我们的剩余工作就是对其进行充实、丰润,完善它的不足之处。

  定义模板采用抽象类来定义,公共的结构化逻辑需要在抽象类中完成,只将非公共的部分逻辑抽象成抽象方法,留待子类充实实现。

  所以上文所述不足之处就是这些抽象方法。

  总的来说,模板模式就是通过抽象类来定义一个逻辑模板,逻辑框架、逻辑原型,然后将无法决定的部分抽象成抽象类交由子类来实现,一般这些抽象类的调用逻辑还是在抽象类中完成的。这么看来,模板就是定义一个框架,比如盖房子,我们定义一个模板:房子要封闭,有门,有窗等等,但是要什么样的门,什么样的窗,这些并不在模板中描述,这个交给子类来完善,比如门使用防盗门,窗使用北向的窗等等。

  我们不凡就以建房为例来见识一下模板模式如何:

模板抽象类:HouseTemplate

 public abstract class HouseTemplate {

     protected HouseTemplate(String name){
this.name = name;
} protected String name; protected abstract void buildDoor(); protected abstract void buildWindow(); protected abstract void buildWall(); protected abstract void buildBase(); //公共逻辑
public final void buildHouse(){ buildBase();
buildWall();
buildDoor();
buildWindow(); } }

子类1:HouseOne

 public class HouseOne extends HouseTemplate {

     HouseOne(String name){
super(name);
} @Override
protected void buildDoor() {
System.out.println(name +"的门要采用防盗门");
} @Override
protected void buildWindow() {
System.out.println(name + "的窗户要面向北方");
} @Override
protected void buildWall() {
System.out.println(name + "的墙使用大理石建造");
} @Override
protected void buildBase() {
System.out.println(name + "的地基使用钢铁地基");
} }

子类2:HouseTwo

 public class HouseTwo extends HouseTemplate {

     HouseTwo(String name){
super(name);
} @Override
protected void buildDoor() {
System.out.println(name + "的门采用木门");
} @Override
protected void buildWindow() {
System.out.println(name + "的窗户要向南");
} @Override
protected void buildWall() {
System.out.println(name + "的墙使用玻璃制造");
} @Override
protected void buildBase() {
System.out.println(name + "的地基使用花岗岩");
} }

测试类:Clienter

 public class Clienter {

     public static void main(String[] args){
HouseTemplate houseOne = new HouseOne("房子1");
HouseTemplate houseTwo = new HouseTwo("房子2");
houseOne.buildHouse();
houseTwo.buildHouse();
} }

测试结果:

房子1的地基使用钢铁地基
房子1的墙使用大理石建造
房子1的门要采用防盗门
房子1的窗户要面向北方
房子2的地基使用花岗岩
房子2的墙使用玻璃制造
房子2的门采用木门
房子2的窗户要向南

执行结果

  通过以上例子,我们认识了模板模式中的基本方法和模板方法,其中HouseTemplate中的buildHouse方法就是基本方法,其余四个均为模板方法。其中基本方法一般会用final修饰,保证其不会被子类修改,而模板方法则使用protected修饰,表明其需要在子类中实现。

  其实,模板模式中还有一个钩子方法的概念,有人称,具有钩子方法的模板模式才算完整,也许吧。

  钩子方法时干啥的呢?钩子就是给子类一个授权,允许子类通过重写钩子方法来颠覆基本逻辑的执行,这有时候是非常有用的。就比如在盖房子的时候,有一个需要子类来决定是否建造厕所间的需求时,可以这么实现:

模板抽象类:HouseTemplate

 public abstract class HouseTemplate {

     protected HouseTemplate(String name){
this.name = name;
} protected String name; protected abstract void buildDoor(); protected abstract void buildWindow(); protected abstract void buildWall(); protected abstract void buildBase(); protected abstract void buildToilet(); //钩子方法
protected boolean isBuildToilet(){
return true;
} //公共逻辑
public final void buildHouse(){ buildBase();
buildWall();
buildDoor();
buildWindow();
if(isBuildToilet()){
buildToilet();
}
} }

子类1:HouseOne

 public class HouseOne extends HouseTemplate {

     HouseOne(String name){
super(name);
} HouseOne(String name, boolean isBuildToilet){
this(name);
this.isBuildToilet = isBuildToilet;
} public boolean isBuildToilet; @Override
protected void buildDoor() {
System.out.println(name +"的门要采用防盗门");
} @Override
protected void buildWindow() {
System.out.println(name + "的窗户要面向北方");
} @Override
protected void buildWall() {
System.out.println(name + "的墙使用大理石建造");
} @Override
protected void buildBase() {
System.out.println(name + "的地基使用钢铁地基");
} @Override
protected void buildToilet() {
System.out.println(name + "的厕所建在东南角");
} @Override
protected boolean isBuildToilet(){
return isBuildToilet;
} }

子类2:HouseTwo

public class HouseTwo extends HouseTemplate {

    HouseTwo(String name){
super(name);
} @Override
protected void buildDoor() {
System.out.println(name + "的门采用木门");
} @Override
protected void buildWindow() {
System.out.println(name + "的窗户要向南");
} @Override
protected void buildWall() {
System.out.println(name + "的墙使用玻璃制造");
} @Override
protected void buildBase() {
System.out.println(name + "的地基使用花岗岩");
} @Override
protected void buildToilet() {
System.out.println(name + "的厕所建在西北角");
} }

测试类:Clienter

 public class Clienter {

     public static void main(String[] args){
HouseTemplate houseOne = new HouseOne("房子1", false);
HouseTemplate houseTwo = new HouseTwo("房子2");
houseOne.buildHouse();
houseTwo.buildHouse();
} }

测试结果:

房子1的地基使用钢铁地基
房子1的墙使用大理石建造
房子1的门要采用防盗门
房子1的窗户要面向北方
房子2的地基使用花岗岩
房子2的墙使用玻璃制造
房子2的门采用木门
房子2的窗户要向南
房子2的厕所建在西北角

执行结果

  通过执行结果我们可以清晰的看到,我们通过重写钩子方法自定义了房子1不需要建造厕所(fasle)。

  钩子方法的作用也就一目了然啦。

  模板模式的关键点:

    1、使用抽象类定义模板类,并在其中定义所有的基本方法、模板方法,钩子方法,不限数量,以实现功能逻辑为主。其中基本方法使用final修饰,其中要调用基本方法和钩子方法,基本方法和钩子方法可以使用protected修饰,表明可被子类修改。

    2、定义实现抽象类的子类,重写其中的模板方法,甚至钩子方法,完善具体的逻辑。

  使用场景:

    1、在多个子类中拥有相同的方法,而且逻辑相同时,可以将这些方法抽出来放到一个模板抽象类中。

    2、程序主框架相同,细节不同的情况下,也可以使用模板方法。


同系列文章:

Java设计模式之《模板模式》及使用场景的更多相关文章

  1. Java设计模式之模板模式及使用场景

    模板模式,顾名思义,就是通过模板拓印的方式. 定义模板,就是定义框架.结构.原型.定义一个我们共同遵守的约定. 定义了模板,我们的剩余工作就是对其进行充实.丰润,完善它的不足之处. 定义模板采用抽象类 ...

  2. 图解Java设计模式之模板模式

    图解Java设计模式之模板模式 豆浆制作问题 模板方法模式基本介绍 模板方法模式原理类图 模板方法模式解决豆浆制作问题 模板方法模式的钩子方法 模板方法模式在Spring框架中的源码分析 模板方法模式 ...

  3. JAVA设计模式之模板模式

    在阎宏博士的<JAVA与模式>一书中开头是这样描述模板方法(Template Method)模式的: 模板方法模式是类的行为模式.准备一个抽象类,将部分逻辑以具体方法以及具体构造函数的形式 ...

  4. Java设计模式之模板模式(Template )

    前言: 最近学习了Glide开源图片缓存框架,在学习到通过使用ModelLoader自定义数据源的时候,Glide巧妙的使用了Java的模板模式来对外暴露处理不同的Url数据源,今天来学习总结一下模板 ...

  5. 13.java设计模式之模板模式

    基本需求: 制作豆浆的流程 选材--->添加配料--->浸泡--->放到豆浆机打碎 通过添加不同的配料,可以制作出不同口味的豆浆 选材.浸泡和放到豆浆机打碎这几个步骤对于制作每种口味 ...

  6. java设计模式之模板模式以及钩子方法使用

    1.使用背景 模板方法模式是通过把不变行为搬到超类,去除子类里面的重复代码提现它的优势,它提供了一个很好的代码复用平台.当不可变和可变的方法在子类中混合在一起的时候, 不变的方法就会在子类中多次出现, ...

  7. Java设计模式之代理模式(静态代理和JDK、CGLib动态代理)以及应用场景

    我做了个例子 ,需要可以下载源码:代理模式 1.前言: Spring 的AOP 面向切面编程,是通过动态代理实现的, 由两部分组成:(a) 如果有接口的话 通过 JDK 接口级别的代理 (b) 如果没 ...

  8. 折腾Java设计模式之模板方法模式

    博客原文地址:折腾Java设计模式之模板方法模式 模板方法模式 Define the skeleton of an algorithm in an operation, deferring some ...

  9. Java设计模式——装饰者模式

    JAVA 设计模式 装饰者模式 用途 装饰者模式 (Decorator) 动态地给一个对象添加一些额外的职责.就增加功能来说,Decorator 模式相比生成子类更为灵活. 装饰者模式是一种结构式模式 ...

  10. JAVA设计模式--装饰器模式

    装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构.这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装饰 ...

随机推荐

  1. yum安装常用工具命令

    树状目录结构(tree) yum install tree 上传下载工具(rz和sz) yum install lrzsz -------------------------------------- ...

  2. MVC+EF 多条件查询

    根据以前的做法是拼接sql语句,这会增加维护成本,因为sql语句里的内容不会报错,所以在使用ef的时候必须要抛弃拼接sql语句的习惯. 构建实例 List<vyw_user> list = ...

  3. cisco基本配置命令

    实验命令 router> enable 从用户模式进入特权模式 router# disable or exit 从特权模式退出到用户模式 router# show sessions 查看本机上的 ...

  4. JavaScriptDOM

    DOM简介 1.HTML DOM:网页被加载时,浏览器会创建文档对象模型 2.DOM操作HTML:改变HTML的元素.属性.CSS样式.对所有事件作出反应 DOM操作HTML 1.改变HTML输出流 ...

  5. SJCP认证题前五十题填坑

    在做Java的SJCP认证试题时自己整理了一些Java基础细节知识点,以下是知识点陈列 1.标签机制:标签起作用的唯一的地方刚好在迭代语句之前(不然编译错误) continue label1 直接转到 ...

  6. 【转】RPC介绍

    转自:http://www.cnblogs.com/Vincentlu/p/4185299.html 摘要: RPC——Remote Procedure Call Protocol,这是广义上的解释, ...

  7. STM32CubeMX+Keil裸机代码风格(1)

    1.打开STM32CubeMX,New project 选好自己要用的芯片 2.选上左侧SYS中的debug Serial Wire(定义烧程序的端口) . 3,选上左侧TIM6,使TIM6可用(TI ...

  8. Java实现对文本文件MD5加密并ftp传送到远程主机目录

    需求描述: 客户出完账之后需要把出账的数据以文本文件的方式传送给收入管理系统,客户以前是通过本地的一个工具软件上传的,由于安全监管的原因,不允许在本地使用工具上传,因此客户希望我们在已经上线使用的系统 ...

  9. vue的学习之路

    一.vs code中,适合vue的前端插件 查看网址:http://blog.csdn.net/caijunfen/article/details/78749766 二.如何使用git从gitub上拉 ...

  10. Rabbitmq的使用及Web监控工具使用

    本文转载自:https://www.cnblogs.com/gossip/p/4475978.html windows安装手册请参考:http://www.rabbitmq.com/install-w ...