设计模式使用场景实现示例及优缺点(行为型模式——模板方法模式)

从前,有一个古老的王国,名叫算法王国。这个王国中,住着一位聪明睿智的国王,他名叫模板国王。模板国王深谙设计的艺术,他不仅是一个杰出的领导者,更是一个伟大的发明家。

有一天,模板国王召集了他的三个孩子,分别是骑士A、骑士B和骑士C。国王告诉他们:“在我们王国中,有许多复杂的任务需要完成。为了更好地管理这些任务,我设计了一种方法,称之为模板方法模式。这是一种行为设计模式,定义了一个操作中的算法骨架,并将算法的一些步骤延迟到你们这些子类中。这样,你们可以在不改变算法结构的前提下,重新定义某些特定步骤。”

骑士们听得一头雾水,国王于是决定用一个简单的例子来解释。他们来到了一片广阔的田野,田野上有许多不同种类的树木。国王指着树木说:“我们要在这片田野上种植更多的树木,但每种树的种植方法有所不同。然而,无论哪种树,都要经历几个基本步骤:准备土壤、播种、浇水、施肥。”

国王继续说道:“现在,我已经制定了一个模板方法,我们称之为‘种树模板’。这个模板包含了基本的四个步骤,但每种树的具体操作可以有所不同。你们每个人负责一种树木,根据模板来执行任务。”

于是,骑士A负责种苹果树,骑士B负责种橡树,骑士C负责种松树。每个骑士都按照模板方法进行操作,但在每个步骤中,根据不同树木的需求,调整了具体的操作。

骑士A在准备土壤时,选择了富含有机质的泥土;在播种时,选择了最佳的季节;浇水时,确保水量适中;施肥时,使用了苹果树专用的肥料。

骑士B在准备土壤时,选择了较为坚硬的土壤;播种时,进行了深度播种;浇水时,每隔几天进行一次大量浇水;施肥时,使用了橡树需要的营养。

骑士C在准备土壤时,选择了砂质土壤;播种时,注意了松树种子的间距;浇水时,采取了滴灌方式;施肥时,选择了适合松树生长的肥料。

通过这种方式,三位骑士不仅完成了种树的任务,还因为各自树木的特点,优化了每个步骤。
通过定义算法的骨架,可以在不改变整体结构的前提下,自由发挥,完成特定的任务。

模板方法模式(Template Method Pattern)

模板方法模式(Template Method Pattern)是一种行为设计模式,它定义了一个操作中的算法的骨架,将算法的一些步骤延迟到子类中。这样可以在不改变算法的结构的前提下,重新定义算法的某些特定步骤。

核心组件

  • AbstractClass(抽象类):这是一个抽象基类,它定义了一套算法的模板。它实现了模板方法,定义了算法的骨架,具体步骤由子类实现。
  • ConcreteClass(具体类):这些类继承自抽象基类,并实现其算法中的具体步骤。

适用场景

  1. 一次性实现算法的不变部分
    • 当算法的大部分结构固定不变,但某些步骤具有多变性时,可使用模板方法模式。
  2. 各子类中公共行为应提取到单一位置避免代码重复
    • 通过模板方法模式,可以将公共行为提取到超类中,通过继承机制复用这部分代码。
  3. 控制子类扩展
    • 模板方法模式通过将算法的核心流程固化在超类中,防止子类破坏原有算法的结构。

实现实例

以一个简单的游戏开发为例,其中游戏的基本结构(启动、开始、结束)是固定的,但具体每个阶段的实现可以根据不同类型的游戏变化。使用模板方法模式可以固定游戏的主体流程,而将具体步骤的实现留给子类:

抽象类(Abstract Class)

这个类定义了游戏的基本流程,并将具体实现留给子类。

public abstract class Game {
    abstract void initialize();
    abstract void startPlay();
    abstract void endPlay();

    // 模板方法
    public final void play() {
        initialize();  // 初始化游戏
        startPlay();   // 开始游戏
        endPlay();     // 结束游戏
    }
}
具体类(Concrete Classes)

这些类继承自抽象基类,并实现了其具体的操作。

public class Cricket extends Game {
    @Override
    void initialize() {
        System.out.println("Cricket Game Initialized! Start playing.");
    }

    @Override
    void startPlay() {
        System.out.println("Cricket Game Started. Enjoy the game!");
    }

    @Override
    void endPlay() {
        System.out.println("Cricket Game Finished!");
    }
}

public class Football extends Game {
    @Override
    void initialize() {
        System.out.println("Football Game Initialized! Start playing.");
    }

    @Override
    void startPlay() {
        System.out.println("Football Game Started. Enjoy the game!");
    }

    @Override
    void endPlay() {
        System.out.println("Football Game Finished!");
    }
}
客户端代码(Client Code)

这部分代码演示了如何使用模板方法模式来规范游戏的流程。

public class Client {
    public static void main(String[] args) {
        Game game = new Cricket();
        game.play();  // 按照Cricket的流程执行游戏

        game = new Football();
        game.play();  // 按照Football的流程执行游戏
    }
}

优缺点

优点
  1. 提高代码复用性
    • 将通用部分的代码放在抽象的父类中,减少了子类的重复代码。
  2. 扩展性好
    • 新增具体类时,只需实现算法的可变部分,不需修改已有的代码。
  3. 控制子类扩展
    • 可以在超类中定义严格的算法规则,限定子类的行为和结构。
缺点
  1. 对继承的依赖
    • 模板方法模式通过继承来实现,可能会导致过多的类层次。
  2. 可能违背Liskov替换原则
    • 如果子类不适当地实现父类的方法,可能会违背Liskov替换原则。

类图

+----------------+         +------------------+
|   AbstractClass|-------->|   ConcreteClass  |
+----------------+         +------------------+
| + templateMethod()       | + step1()        |
| + step1()                | + step2()        |
| + step2()                +------------------+
| + step3()                |
+----------------+         |
| + step3()                |
+----------------+         |
                           |
      +-------------------+|
      |                   |
+---------------+ +-----------------+
|ConcreteClassA | |ConcreteClassB   |
+---------------+ +-----------------+
| + step1()     | | + step1()       |
| + step2()     | | + step2()       |
| + step3()     | | + step3()       |
+---------------+ +-----------------+

注意事项

设计灵活性与复杂性:

模板方法模式虽然提高了代码的复用性,但也可能导致设计过于复杂。在设计时应确保不过度使用,以免造成系统的不必要复杂。

子类的设计约束:

子类实现时必须遵循抽象基类的方法模板,这限制了子类的灵活性。设计者需要在提供足够的灵活性和维持算法结构之间找到平衡。

重构与维护:

如果模板方法本身需要修改,可能会影响到所有的子类。因此,在模板方法中应尽量减少修改的可能性,确保长时间的稳定性。

总结

模板方法模式是一种强大的设计工具,通过预定义算法的结构,提供了高度的复用性和扩展性。它不仅适用于软件开发,还广泛应用于系统设计和业务流程管理。通过对模式的适当扩展和优化,可以有效应对更加复杂和动态的设计挑战。这种模式的成功实施需要深入理解业务需求和技术上的灵活运用,以确保设计的可维护性和系统的可扩展性。

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

懒人w

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值