基本需求:

  • 制作豆浆的流程 选材--->添加配料--->浸泡--->放到豆浆机打碎
  • 通过添加不同的配料,可以制作出不同口味的豆浆
  • 选材、浸泡和放到豆浆机打碎这几个步骤对于制作每种口味的豆浆都是一样的
  • 通过模板方法可以完成

基本介绍:

  • 模板方法模式(Template Method),又叫模板模式(Template),在一个抽象类公开定义了执行它的方法的模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行

  • 模板方法模式 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中,使得子类可以不改变一个算法的结构,就可以重定义该算法的某些特定步骤,这种类型的设计模式属于行为型模式

  • UML类图(原理)

    • 说明

      • AbstractClass:抽象类,其中的template()为模板方法

        • public final void template() {
          // 在模板方法内部调用其他方法 规定算法的具体流程,这些方法可以是抽象的,也可以不是抽象的
          // 抽象的方法就交由子类实现,子类不改变算法的结构重定义算法的某些步骤
          // 模板方法内部调用自身实现的方法,可作为钩子方法,让子类有选择的去实现
          operation1();
          operation2();
          }
      • ConcreationClass:实现类,对template()方法内部所使用的方法进行实现,对钩子方法可以选择性的实现,不实现则使用父类的实现

  • UML类图(案例)

  • 代码实现

    • public abstract class SoyMilk {
      
         // 模板方法 使用final 不能让子类重写,定义好具体的算法流程,其实现子类去实现
      public final void make() {
      select();
      if (isAdd()) {
      // 使用钩子方法判断是否需要加配料 默认加
      add();
      }
      soak();
      beat();
      } // 添加配料方法 ,不同品种的豆浆,加的配料不一样
      public abstract void add(); // 该方法为钩子方法,子类可根据情况是否实现该方法
      public boolean isAdd() {
      return true;
      } public void select() {
      System.out.println("选择好的新鲜黄豆");
      } public void soak() {
      System.out.println("黄豆和配料开始浸泡,需要3小时");
      } public void beat() {
      System.out.println("黄豆和配料放到豆浆机去打碎");
      } } // 子类一
      class RedBeanSoyMilk extends SoyMilk {
      // 重写添加方法
      @Override
      public void add() {
      System.out.println("添加上好的红豆");
      } } // 子类二
      class PeanutSoyMilk extends SoyMilk {
      // 重写添加方法
      @Override
      public void add() {
      System.out.println("添加上好的花生");
      } } // 子类三
      class PureSoyMilk extends SoyMilk {
      // 重写添加方法 和 钩子方法
      @Override
      public void add() {
      System.out.println("添加上好的花生");
      } @Override
      public boolean isAdd() {
      return false;
      }
      }
    • public class Client {
      public static void main(String[] args) {
      // 制作红豆豆浆 调用模板方法
      SoyMilk redBeanSoyMilk = new RedBeanSoyMilk();
      redBeanSoyMilk.make();
      // 制作花生豆浆 调用模板方法
      SoyMilk peanutSoyMilk = new PeanutSoyMilk();
      peanutSoyMilk.make();
      // 制作纯豆浆 重写了钩子方法 调用模板方法
      SoyMilk pureSoyMilk = new PureSoyMilk();
      pureSoyMilk.make();
      }
      }

spring源码:

  • 在spring的IOC容器中就使用到了模板模式
  • UML类图
    • 说明
      • refresh()方法调用该类很多方法,也就是模板方法
      • obtainFreshBeanFactory()方法内部调用了refreshBeanFactory()和getBeanFactory()方法,这两个方法是抽象方法,交由了子类去完成
      • postProcessBeanFactory()和onRefresh()方法为钩子方法,在AbstractApplicationContext()内部进行了空实现,子类可以有选择的进行重写

注意事项:

  • 基本思想是:算法只存在于一个地方,也就是在父类中,容易修改。需要修改算法时,只要修改父类的模板方法或者已经实现的某些步骤,子类就会继承这些修改
  • 实现了最大化代码复用。父类的模板方法和已实现的某些步骤会被子类继承而直接使用
  • 既统一了算法,也提供了很大的灵活性。父类的模板方法确保了算法的结构保持不变,同时由子类提供部分步骤的实现
  • 该模式的不足之处:每一个不同的实现都需要一个子类实现,导致类的个数增加,使得系统更加庞大
  • 一般模板方法都加上 final 关键字, 防止子类重写模板方法
  • 模板方法模式使用场景:当要完成在某个过程,该过程要执行一系列步骤 ,这一系列的步骤基本相同,但其个别步骤在实现时可能不同,通常考虑用模板方法模式来处理

13.java设计模式之模板模式的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

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

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

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

  8. 浅析JAVA设计模式之工厂模式(一)

    1 工厂模式简单介绍 工厂模式的定义:简单地说,用来实例化对象,取代new操作. 工厂模式专门负责将大量有共同接口的类实例化.工作模式能够动态决定将哪一个类实例化.不用先知道每次要实例化哪一个类. 工 ...

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

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

随机推荐

  1. Vue中封装axios组件实例

    首先要创建一个网络模块network文件夹  里面要写封装好的几个组件 在config.js里面这样写 在index.js要这样写 core.js文件里面内容如下 然后要在main.js文件里面要设置 ...

  2. JS的各种数据类型

    Number js与其他编程不一样,不管是整数还是浮点,都称为数字类型(Number) 例:123,1.11111,-960 当该类型结果不存在时,即表示为 NaN (Not a Number) In ...

  3. Azure Kay Vault(一).NET Core Console App 获取密钥保管库中的机密信息

    一,引言 Azure 密钥保管库用于存储敏感信息,例如链接字符串,密码,API 密钥等.我们无法直接从Azure 密钥库中访问机密!那么我们如何才能访问应用程序中的机密信息?比如,在我们的实际项目中, ...

  4. SaaS系统怎么做物流行业年度经营报告,MVC+js+echarts实现

    前言 马上就到年底了,很多公司都要汇总这一年的经营情况,如果一个系统没有自动生成年报的功能, 需要人工手工去做年报,我相信可能是一个不小的工作量,最近我通过一个星期的时间,结合系统情况自动生成年报,全 ...

  5. Qlik Sense插件及QRS接口补充

    date: 2019-10-18 09:10:00 updated: 2019-10-18 15:18:00 Qlik Sense插件及QRS接口补充 1.插件 1.1 获取数据方式 理论上 Engi ...

  6. matplotlib中plt用法实例

    import torch from models.models import Model import cv2 from PIL import Image import numpy as np fro ...

  7. D. 停不下来的团长奥尔加 动态规划

    题目描述 分析 设\(f[i]\) 为从 \(i\) 走到 \(i+1\) 的步数 初始值 \(f[i]=2\) 则 \(f[i]=\sum_{i=p[i]}^{i}f[i]\) 考试的时候用树状数组 ...

  8. Bootstrap Blazor 初体验

    自微软去年发布blazor以来,我也一直关注着blazor的动态,从net core 3.1 到 net 5,从 server side 到 wasm client , 点点滴滴印证了 blazor ...

  9. SQL SERVER级联查询及数据结构《存储过程-递归树形查询》

    --创建表,插入数据 create table tb(id varchar(3) , pid varchar(3) , name varchar(10))insert into tb values(' ...

  10. 记EF的一个基本访问类

    代码: 1 using EFModel; 2 using System; 3 using System.Collections.Generic; 4 using System.Data.Entity; ...