【设计模式】模版方法模式

模板方法模式

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

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

  这样的例子在生活中还有很多,例如,一个人每天会起床、吃饭、做事、睡觉等,其中“做事”的内容每天可能不同。我们把这些规定了流程或格式的实例定义成模板,允许使用者根据自己的需求去更新它,例如,简历模板、论文模板、Word 中模板文件等。

  以下介绍的模板方法模式将解决以上类似的问题。

举个现实中的例子

豆浆制作问题

编写制作豆浆的程序,说明如下:

  1. 制作豆浆的流程选材--->添加配料--->浸泡--->放到豆浆机打碎

  2. 通过添加不同的配料,可以制作出不同口味的豆浆

  3. 选材、浸泡和放到豆浆机打碎这几个步骤对于制作每种口味的豆浆都是一样的

  4. 请使用模板方法模式完成(说明:因为模板方法模式,比较简单,很容易就想到这个方案,因此就直接使用,不再使用传统的方案来引出模板方法模式)

模板方法模式基本介绍

基本介绍

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

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

  3. 这种类型的设计模式属于行为型模式

该模式的主要优点如下:

  1. 它封装了不变部分,扩展可变部分。它把认为是不变部分的算法封装到父类中实现,而把可变部分算法由子类继承实现,便于子类继续扩展

  2. 它在父类中提取了公共的部分代码,便于代码复用

  3. 部分方法是由子类实现的,因此子类可以通过扩展方式增加相应的功能,符合开闭原则

该模式的主要缺点如下:

  1. 对每个不同的实现都需要定义一个子类,这会导致类的个数增加,系统更加庞大,设计也更加抽象

  2. 父类中的抽象方法由子类实现,子类执行的结果会影响父类的结果,这导致一种反向的控制结构,它提高了代码阅读的难度

模式的结构与实现

  模板方法模式需要注意抽象类与具体子类之间的协作。它用到了虚函数的多态性技术以及“不用调用我,让我来调用你”的反向控制技术。现在来介绍它们的基本结构。

模式的结构

模板方法模式包含以下主要角色:

(1) 抽象类(Abstract Class):负责给出一个算法的轮廓和骨架。它由一个模板方法和若干个基本方法构成。

这些方法的定义如下:

  ① 模板方法:定义了算法的骨架,按某种顺序调用其包含的基本方法。

  ② 基本方法:是整个算法中的一个步骤,包含以下几种类型。

    • 抽象方法:在抽象类中申明,由具体子类实现。
    • 具体方法:在抽象类中已经实现,在具体子类中可以继承或重写它。
    • 钩子方法:在抽象类中已经实现,包括用于判断的逻辑方法和需要子类重写的空方法两种。

(2) 具体子类(Concrete Class):实现抽象类中所定义的抽象方法和钩子方法,它们是一个顶级逻辑的一个组成步骤。

结构图如下

  

模式的实现

  模板方法模式的代码如下:

 1 public class TemplateDemo {
 2     public static void main(String[] args) {
 3         // 制作红豆豆浆
 4         System.out.println("----制作红豆豆浆----");
 5         SoyaMilk redBeanSoyaMilk = new RedBeanSoyaMilk();
 6         redBeanSoyaMilk.make();
 7         System.out.println("----制作花生豆浆----");
 8         SoyaMilk peanutSoyaMilk = new PeanutSoyaMilk();
 9         peanutSoyaMilk.make();
10     }
11 }
12 
13 // 抽象类,表示豆浆
14 abstract class SoyaMilk {
15 
16     // 模板方法,make,模板方法可以做成 final,不让子类去覆盖
17     final void make() {
18         select();
19         addCondiments();
20         soak();
21         beat();
22     }
23 
24     // 选材料
25     void select() {
26         System.out.println("第一步:选择好的新鲜黄豆 ");
27     }
28 
29     // 添加不同的配料,抽象方法,子类具体实现
30     abstract void addCondiments();
31 
32     // 浸泡
33     void soak() {
34         System.out.println("第三步,黄豆和配料开始浸泡,需要 3 小时 ");
35     }
36 
37     // 打碎
38     void beat() {
39         System.out.println("第四步:黄豆和配料放到豆浆机去打碎 ");
40     }
41 }
42 
43 
44 // 花生豆浆
45 class PeanutSoyaMilk extends SoyaMilk {
46 
47     @Override
48     void addCondiments() {
49         System.out.println(" 加入上好的花生 ");
50     }
51 }
52 
53 // 红豆豆浆
54 class RedBeanSoyaMilk extends SoyaMilk {
55 
56     @Override
57     void addCondiments() {
58         System.out.println(" 加入上好的红豆 ");
59     }
60 }

模板方法模式的钩子方法

  1. 在模板方法模式的父类中,我们可以定义一个方法,它默认不做任何事,子类可以视情况要不要覆盖它,该方法称为“钩子”

  2. 还是用上面做豆浆的例子来讲解,比如,我们希望制作纯豆浆,不添加任何的配料,请使用钩子方法对前面的模板方法进行改造

 1 package com.test.template.improve;
 2 
 3 public class TemplateDemo2 {
 4 
 5     public static void main(String[] args) {
 6         // 制作红豆豆浆
 7         System.out.println("----制作红豆豆浆----");
 8         SoyaMilk redBeanSoyaMilk = new RedBeanSoyaMilk();
 9         redBeanSoyaMilk.make();
10         System.out.println("----制作花生豆浆----");
11         SoyaMilk peanutSoyaMilk = new PeanutSoyaMilk();
12         peanutSoyaMilk.make();
13     }
14 }
15 
16 // 抽象类,表示豆浆
17 abstract class SoyaMilk {
18 
19     // 模板方法,make,模板方法可以做成 final,不让子类去覆盖
20     final void make() {
21         select();
22         if (customerWantCondiments()) {
23             addCondiments();
24         }
25         soak();
26         beat();
27     }
28 
29     // 选材料
30     void select() {
31         System.out.println("第一步:选择好的新鲜黄豆 ");
32     }
33 
34     // 添加不同的配料,抽象方法,子类具体实现
35     abstract void addCondiments();
36 
37     // 浸泡
38     void soak() {
39         System.out.println("第三步,黄豆和配料开始浸泡,需要 3 小时 ");
40     }
41 
42     // 打碎
43     void beat() {
44         System.out.println("第四步:黄豆和配料放到豆浆机去打碎 ");
45     }
46 
47     // 钩子方法,决定是否需要添加配料
48     boolean customerWantCondiments() {
49         return true;
50     }
51 }
52 
53 
54 // 花生豆浆
55 class PeanutSoyaMilk extends SoyaMilk {
56 
57     @Override
58     void addCondiments() {
59         System.out.println(" 加入上好的花生 ");
60     }
61 }
62 
63 // 红豆豆浆
64 class RedBeanSoyaMilk extends SoyaMilk {
65 
66     @Override
67     void addCondiments() {
68         System.out.println(" 加入上好的红豆 ");
69     }
70 }
71 
72 // 纯豆豆浆
73 class PureSoyaMilk extends SoyaMilk{
74     @Override
75     void addCondiments() {
76         //空实现
77     }
78 
79     @Override
80     boolean customerWantCondiments() {
81         return false;
82     }
83 }

模板方法模式在 Spring 框架应用的源码分析

  1. Spring IOC 容器初始化时运用到的模板方法模式

  2. 代码分析+角色分析+说明类图

  

  

模式的应用场景 

模板方法模式通常适用于以下场景:

  1. 算法的整体步骤很固定,但其中个别部分易变时,这时候可以使用模板方法模式,将容易变的部分抽象出来,供子类实现

  2. 当多个子类存在公共的行为时,可以将其提取出来并集中到一个公共父类中以避免代码重复。首先,要识别现有代码中的不同之处,并且将不同之处分离为新的操作。最后,用一个调用这些新的操作的模板方法来替换这些不同的代码

  3. 当需要控制子类的扩展时,模板方法只在特定点调用钩子操作,这样就只允许在这些点进行扩展

模板方法模式的注意事项和细节  

  1. 基本思想是:算法只存在于一个地方,也就是在父类中,容易修改。需要修改算法时,只要修改父类的模板方法或者已经实现的某些步骤,子类就会继承这些修改

  2. 实现了最大化代码复用。父类的模板方法和已实现的某些步骤会被子类继承而直接使用

  3. 既统一了算法,也提供了很大的灵活性。父类的模板方法确保了算法的结构保持不变,同时由子类提供部分步骤的实现

  4. 该模式的不足之处:每一个不同的实现都需要一个子类实现,导致类的个数增加,使得系统更加庞大

  5. 一般模板方法都加上final关键字,防止子类重写模板方法

  6. 模板方法模式使用场景:当要完成在某个过程,该过程要执行一系列步骤,这一系列的步骤基本相同,但其个别步骤在实现时可能不同,通常考虑用模板方法模式来处理

原文链接:https://blog.csdn.net/qq784515681/article/details/106679573

原文地址:https://www.cnblogs.com/h--d/p/14576956.html