唐宋元明清 唐宋元明清
首页
  • 基础

    • Java基础
  • 工具

    • hutool
    • commons
  • 框架

    • Spring Boot相关
  • 设计模式

    • 设计模式入门
  • Hadoop

    • Hadoop分布式搭建
    • Hadoop高可用搭建
    • 集群端口
    • 代码demo
  • Zookeeper

    • Zookeeper集群搭建
  • Hive

    • Hive集群搭建
    • Hive相关
    • HSQL
  • Kafka

    • Kafka集群搭建
  • HBase

    • HBase集群搭建
    • HBase基础学习
  • Spark

    • Spark环境搭建
    • Spark相关知识
  • Flink

    • Flink环境搭建
    • Flink学习
  • Flume

    • Flume安装配置
    • Flume高可用集群安装
    • Flume相关学习
  • Sqoop

    • Sqoop安装配置
    • Sqoop使用
  • 其他

    • docker
  • Oracle

    • Oracle相关知识杂记
    • 系统函数篇
    • 与MySQL语法区别
  • MySQL

    • MySQL知识点
  • Python

    • Python简单语法
    • Python操作Office
    • Python类库学习
    • Python爬虫
  • Shell

    • Shell基础
    • Shell命令行
  • Scala

    • 语法学习
  • 正则表达式

    • 正则基础
  • 调度

    • 调度工具
  • 前端

    • 前端相关
  • 杂记

    • 常用工具或网站
    • 琐碎知识
  • 摘录

    • 摘录
GitHub (opens new window)
首页
  • 基础

    • Java基础
  • 工具

    • hutool
    • commons
  • 框架

    • Spring Boot相关
  • 设计模式

    • 设计模式入门
  • Hadoop

    • Hadoop分布式搭建
    • Hadoop高可用搭建
    • 集群端口
    • 代码demo
  • Zookeeper

    • Zookeeper集群搭建
  • Hive

    • Hive集群搭建
    • Hive相关
    • HSQL
  • Kafka

    • Kafka集群搭建
  • HBase

    • HBase集群搭建
    • HBase基础学习
  • Spark

    • Spark环境搭建
    • Spark相关知识
  • Flink

    • Flink环境搭建
    • Flink学习
  • Flume

    • Flume安装配置
    • Flume高可用集群安装
    • Flume相关学习
  • Sqoop

    • Sqoop安装配置
    • Sqoop使用
  • 其他

    • docker
  • Oracle

    • Oracle相关知识杂记
    • 系统函数篇
    • 与MySQL语法区别
  • MySQL

    • MySQL知识点
  • Python

    • Python简单语法
    • Python操作Office
    • Python类库学习
    • Python爬虫
  • Shell

    • Shell基础
    • Shell命令行
  • Scala

    • 语法学习
  • 正则表达式

    • 正则基础
  • 调度

    • 调度工具
  • 前端

    • 前端相关
  • 杂记

    • 常用工具或网站
    • 琐碎知识
  • 摘录

    • 摘录
GitHub (opens new window)
  • Java相关知识
  • 基础

    • Java基础
    • Java扩展
  • 工具

    • hutool工具包
    • Commons类库
    • SQL解析工具
  • 框架

    • Spring Boot相关
  • 设计模式

    • 设计模式入门
    • 策略模式
    • 观察者模式
    • 装饰者模式
    • 工厂模式
    • 单例模式
    • 命令模式
    • 适配器模式与外观模式
    • 模板方法模式
      • 模板方法模式(Template Method Pattern)
        • 快速写出咖啡和茶的类
        • 第一版设计
        • 类图
        • 代码实现(挂钩)
        • 好莱坞原则
        • Java Api 中的模板方法模式
        • 要点
    • 迭代器与组合模式
    • State 模式
    • 代理模式
  • Java相关
  • 设计模式
Ai
2022-07-25
目录

模板方法模式

# 模板方法模式(Template Method Pattern)

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

这个模式是用来创建一个算法的模板。什么是模板?如你所见的,模板就是一个方法。更具体地说,这个方法将算法定义成一组步骤,其中的任何步骤都可以是抽象的,由子类负责实现。这可以确保算法的结构保持不变,同时由子类提供部分实现。

# 快速写出咖啡和茶的类


    class Coffee {
        void prepareRecipe() {
            boilWater();
            brewCoffeeGrinds();
            pourInCup();
            addSugerAndMilk();
        }
    
        public void boilWater() {
            System.out.println("把水煮沸");
        }
        public void brewCoffeeGrinds() {
            System.out.println("用沸水冲泡咖啡");
        }
        public void pourInCup() {
            System.out.println("吧咖啡倒进杯子");
        }
        public void addSugerAndMilk() {
            System.out.println("加糖和牛奶");
        }
    }
    
    public class Tea {
        void prepareRecipe() {
            boilWater();
            steepTeaBag();
            pourInCup();
            addLemon();
        }
    
        public void boilWater() {
            System.out.println("把水煮沸");
        }
        public void steepTeaBag() {
            System.out.println("用沸水浸泡茶叶");
        }
        public void pourInCup() {
            System.out.println("把茶倒进杯子");
        }
        public void addLemon() {
            System.out.println("加柠檬");
        }
    }
    
    // Make sure to add code blocks to your code group

    # 第一版设计

    1. prepareRecipe() 方法在每个类中都不一样,所以定义为抽象方法;每个子类都覆盖 prepareRecipe(),实现自己的冲泡法
    2. boilWater() 和 pourInCup() 为相同的方法,所以定义在超类中;特有方法放到子类中

    # 类图

    1. CaffeineBeverage 是高层组件,他能控制冲泡法的算法,只有在需要子类实现某个方法时,才调用子类
    2. 饮料的客户代码只依赖 CaffeineBeverage 抽象,而不依赖具体的 Tea 或者 Coffee 这可以减少整个系统的依赖
    3. 子类只简单用来提供一些实现细节
    4. Tea 和 Coffee 如果没有先被调用,绝对不会调用抽象类

    # 代码实现(挂钩)


      /**
       * 抽象类,用来作为基类,其子类必须实现其操作
       */
      abstract class CaffeineBeverageWithHook {
          /**
           * 模板方法被声明为 final,以免算法顺序被改变(不能被子类覆盖)
           * 此方法定义了一连串步骤,每个步骤为一个方法
           */
          final void prepareRecipe() {
              boilWater();
              brew();
              pourInCup();
              // 加上条件语句,条件成立(customerWantsCondiments() 为 true,即顾客想要调料 )时,才调用 addCondiments() 方法
              if (customerWantsCondiments()) {
                  addCondiments();
              }
          }
      
          /**
           * 原语操作,子类必须实现他们
           */
          abstract void brew();
          abstract void addCondiments();
      
          void boilWater() {
              System.out.println("把水煮沸");
          }
          void pourInCup() {
              System.out.println("倒进杯子中");
          }
      
          /**
           * 这就是一个钩子,子类有可能覆盖此方法
           * @return
           */
          boolean customerWantsCondiments () {
              return true;
          }
      
      }
      
      class CoffeeWithHook extends CaffeineBeverageWithHook {
      
          @Override
          public void brew() {
              System.out.println("用沸水冲泡咖啡");
          }
      
          @Override
          public void addCondiments() {
              System.out.println("加糖和牛奶");
          }
      
          /**
           * 使用钩子:覆盖钩子,提供自己的功能
           * @return
           */
          @Override
          public boolean customerWantsCondiments() {
              String answer = getUserInput();
              return "y".equalsIgnoreCase(answer);
          }
      
          private String getUserInput () {
              String answer = null;
      
              System.out.println("咖啡里要加糖和牛奶吗?(y/n)");
      
              BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in));
              try {
                  answer = bufferedReader.readLine();
              } catch (IOException e) {
                  e.printStackTrace();
              } finally {
                  if (null != bufferedReader) {
                      try {
                          bufferedReader.close();
                      } catch (IOException e) {
                          e.printStackTrace();
                      }
                  }
              }
      
              if (null == answer) answer = "n";
              return answer;
          }
      }
      
      public class TeaWithHook extends CaffeineBeverageWithHook {
      
          @Override
          public void brew() {
              System.out.println("用沸水浸泡茶叶");
          }
      
          @Override
          public void addCondiments() {
              System.out.println("加柠檬");
          }
      }
      
      public class Main {
          public static void main(String[] args) {
              TeaWithHook teaWithHook = new TeaWithHook();
              CoffeeWithHook coffeeWithHook = new CoffeeWithHook();
      
              System.out.println("制作茶...");
              teaWithHook.prepareRecipe();
      
              System.out.println("制作咖啡...");
              coffeeWithHook.prepareRecipe();
          }
      }
      
      // Make sure to add code blocks to your code group

      # 好莱坞原则

      别调用(CALL)我们,我们会调用(CALL)你

      好菜坞原则可以给我们一种防止“依赖腐败”的方法。当高层组件依赖低层组件,而低层组件又依赖高层组件,而高层组件又依赖边侧组件,而边侧组件又依赖低层组件时,依赖腐败就发生了。在这种情况下,没有人可以轻易地搞懂系统是如何设计的。

      jxNHmD.png (opens new window)

      # Java Api 中的模板方法模式

      1. 模板方法排序(java.util.Arrays 的 sort 方法)

      比较类实现 Comparable 接口并实现 compareTo 方法,就可以使用 Arrays.sort() 进行排序

      1. Swing 的 JFrame(其中 paint() 方法是一个钩子)
      2. Applet(具体的applet大量使用钩子来提供行为。因为这些行为是作为钩子实现的,所以Applet类就不用去实现它们。)
        • init 钩子用来进行 applet 的初始化动作,会在 applet 一开始的时候调用一次
        • repaint() 是一个具体方法,可让上层组件知道这个 applet 要重绘
        • start 钩子可以在 applet 正要显示在网页上时,让 applet 做一些事情
        • 若跳到别的网页 stop() 钩子会被调用
        • applet 即将被销毁,destroy 钩子会被调用

      # 要点

      • “模板方法”定义了算法的步骤,把这些步骤的实现延迟到子类。
      • 模板方法模式为我们提供了一种代码复用的重要技巧。
      • 模板方法的抽象类可以定义具体方法、抽象方法和钩子。
      • 抽象方法由子类实现。
      • 钩子是一种方法,它在抽象类中不做事,或着只做默认的事情,子类可以选择要不要去覆盖它。
      • 为了防止子类改变模板方法中的算法,可以将模板方法声明为 final。
      • 好莱坞原则告诉我们,将决策权放在高层模块中,以便决定如何以及何时调用低层模块。
      • 你将在真实世界代码中看到模板方法模式的许多变体,不要期望一眼就可以认出它们。
      • 策略模式和模板方法模式都封装算法,一个用组合,一个用继承。
      • 工厂方法是模板方法的一种特殊饭本。
      编辑 (opens new window)
      上次更新: 2022/07/26, 00:03:35
      适配器模式与外观模式
      迭代器与组合模式

      ← 适配器模式与外观模式 迭代器与组合模式→

      Theme by Vdoing | Copyright © 2022-2025 Ai | MIT License
      • 跟随系统
      • 浅色模式
      • 深色模式
      • 阅读模式
      ×