首页 > 编程语言 >程序员必知!模板方法模式的实战应用与案例分析

程序员必知!模板方法模式的实战应用与案例分析

时间:2024-01-12 10:02:03浏览次数:41  
标签:... 咖啡 必知 void 咖啡豆 程序员 制作 模板

Java核心基础 - 程序员古德

模板方法模式让子类在不改变算法整体结构的前提下定制特定步骤,例如咖啡制作,不同咖啡遵循相同流程但有独特定制,如拿铁加牛奶,美式不加,这确保了制作流程的一致性,同时满足了不同咖啡的个性化需求,体现了模板方法模式的核心思想。

定义

程序员必知!模板方法模式的实战应用与案例分析 - 程序员古德

模板方法模式是一种行为设计模式,它在一个方法中定义了一个算法的骨架,允许子类在不改变算法结构的情况下,对算法的某些步骤中的内容进行定制。

举一个现实中的例子:假设有一家咖啡店,店中提供多种咖啡,比如拿铁、美式、卡布奇诺等,虽然这些咖啡的制作过程有所不同,但它们都遵循一个基本的制作流程:准备咖啡豆、研磨咖啡豆、冲泡咖啡、倒入杯子、加入配料(如牛奶、糖等)、完成制作。

在这个例子中,制作咖啡的流程就是一个模板方法,它定义了一个算法的骨架,不同的咖啡(拿铁、美式、卡布奇诺等)就是这个模板方法的具体实现,它们会在算法的某些步骤上进行定制,比如,拿铁咖啡在冲泡后会加入牛奶,而美式咖啡则不会;卡布奇诺咖啡可能会在冲泡前多加入一些研磨后的咖啡豆,以增强咖啡的口感。

通过这种方式,可以确保每一种咖啡的制作都遵循基本的流程,同时又可以根据具体咖啡的需求对流程中的某些步骤进行定制,这就是模板方法的核心思想。

代码案例

程序员必知!模板方法模式的实战应用与案例分析 - 程序员古德

正例

下面是一个未使用模板方法的反例代码,在这个例子中,假设有一个咖啡制作系统,其中包含了制作不同种类咖啡的方法,但由于没有使用模板方法,这些方法之间会存在大量的重复代码。

先定义一个CoffeeMaker类,它包含了制作拿铁咖啡和美式咖啡的方法,如下代码:

public class CoffeeMaker {  
  
    // 制作拿铁咖啡  
    public void makeLatte() {  
        prepareCoffeeBeans();  
        grindCoffeeBeans();  
        brewCoffee();  
        pourIntoCup();  
        addMilk(); // 拿铁咖啡特有的步骤:加入牛奶  
        System.out.println("拿铁咖啡制作完成!");  
    }  
  
    // 制作美式咖啡  
    public void makeAmericanCoffee() {  
        prepareCoffeeBeans();  
        grindCoffeeBeans();  
        brewCoffee();  
        pourIntoCup();  
        // 美式咖啡不需要添加其他配料  
        System.out.println("美式咖啡制作完成!");  
    }  
  
    // 准备咖啡豆  
    private void prepareCoffeeBeans() {  
        System.out.println("准备咖啡豆...");  
    }  
  
    // 研磨咖啡豆  
    private void grindCoffeeBeans() {  
        System.out.println("研磨咖啡豆...");  
    }  
  
    // 冲泡咖啡  
    private void brewCoffee() {  
        System.out.println("冲泡咖啡...");  
    }  
  
    // 倒入杯子  
    private void pourIntoCup() {  
        System.out.println("倒入杯子...");  
    }  
  
    // 拿铁咖啡特有的步骤:加入牛奶  
    private void addMilk() {  
        System.out.println("加入牛奶...");  
    }  
}

接下来,编写一个client类CoffeeShop,用于调用CoffeeMaker中的方法来制作咖啡,如下代码:

public class CoffeeShop {  
    public static void main(String[] args) {  
        CoffeeMaker coffeeMaker = new CoffeeMaker();  
  
        // 制作拿铁咖啡  
        coffeeMaker.makeLatte();  
  
        // 制作美式咖啡  
        coffeeMaker.makeAmericanCoffee();  
    }  
}

运行CoffeeShopmain方法,输出将会是,如下代码:

准备咖啡豆...  
研磨咖啡豆...  
冲泡咖啡...  
倒入杯子...  
加入牛奶...  
拿铁咖啡制作完成!  
准备咖啡豆...  
研磨咖啡豆...  
冲泡咖啡...  
倒入杯子...  
美式咖啡制作完成!

从上面的代码中,可以看到makeLattemakeAmericanCoffee方法中有很多重复的步骤,如准备咖啡豆、研磨咖啡豆、冲泡咖啡和倒入杯子,这就是没有使用模板方法所带来的问题,当需要修改这些公共步骤时,必须在每个方法中都进行修改,这增加了出错的可能性和维护的难度。

如果使用了模板方法,可以将这些公共步骤提取到一个模板方法中,然后让具体的咖啡类(如拿铁咖啡、美式咖啡)继承这个模板方法,并只重写它们特有的步骤,从而减少重复代码并提高系统的灵活性。

反例

下面是一个使用模板方法的正例代码,在这个例子中,定义一个咖啡制作的模板方法,并让具体的咖啡类(如拿铁咖啡、美式咖啡)继承并实现它们特有的步骤。

首先,定义一个抽象类AbstractCoffee,它包含了制作咖啡的模板方法以及一些公共步骤的实现,如下代码:

// 抽象咖啡类,定义制作咖啡的模板方法  
abstract class AbstractCoffee {  
  
    // 模板方法,定义制作咖啡的算法骨架  
    public final void prepareCoffee() {  
        prepareCoffeeBeans();  
        grindCoffeeBeans();  
        brewCoffee();  
        pourIntoCup();  
        addIngredients(); // 由子类实现的具体步骤  
        System.out.println(getCoffeeName() + "制作完成!");  
    }  
  
    // 准备咖啡豆  
    public void prepareCoffeeBeans() {  
        System.out.println("准备咖啡豆...");  
    }  
  
    // 研磨咖啡豆  
    public void grindCoffeeBeans() {  
        System.out.println("研磨咖啡豆...");  
    }  
  
    // 冲泡咖啡,抽象方法,由子类实现  
    protected abstract void brewCoffee();  
  
    // 倒入杯子  
    public void pourIntoCup() {  
        System.out.println("倒入杯子...");  
    }  
  
    // 添加配料,钩子方法,子类可以覆盖实现也可以不实现  
    protected void addIngredients() {  
        // 默认不添加任何配料  
    }  
  
    // 获取咖啡名称,抽象方法,由子类实现  
    protected abstract String getCoffeeName();  
}

接下来,定义具体的咖啡类,继承自AbstractCoffee并实现它们特有的步骤,如下代码:

// 拿铁咖啡类  
class Latte extends AbstractCoffee {  
  
    @Override  
    protected void brewCoffee() {  
        System.out.println("冲泡拿铁咖啡...");  
    }  
  
    @Override  
    protected void addIngredients() {  
        System.out.println("加入牛奶...");  
    }  
  
    @Override  
    protected String getCoffeeName() {  
        return "拿铁咖啡";  
    }  
}  
  
// 美式咖啡类  
class AmericanCoffee extends AbstractCoffee {  
  
    @Override  
    protected void brewCoffee() {  
        System.out.println("冲泡美式咖啡...");  
    }  
  
    @Override  
    protected String getCoffeeName() {  
        return "美式咖啡";  
    }  
}

最后,编写一个client类CoffeeShop,用于调用具体的咖啡类来制作咖啡,如下代码:

// 咖啡店客户端类  
public class CoffeeShop {  
    public static void main(String[] args) {  
        // 制作拿铁咖啡  
        AbstractCoffee latte = new Latte();  
        latte.prepareCoffee();  
  
        // 制作美式咖啡  
        AbstractCoffee americanCoffee = new AmericanCoffee();  
        americanCoffee.prepareCoffee();  
    }  
}

运行CoffeeShopmain方法,输出将会是:

准备咖啡豆...  
研磨咖啡豆...  
冲泡拿铁咖啡...  
倒入杯子...  
加入牛奶...  
拿铁咖啡制作完成!  
准备咖啡豆...  
研磨咖啡豆...  
冲泡美式咖啡...  
倒入杯子...  
美式咖啡制作完成!

在这个例子中,使用了模板方法来定义咖啡制作的流程,通过在抽象类AbstractCoffee中定义模板方法prepareCoffee,确保了所有咖啡的制作都遵循相同的步骤,同时允许子类(拿铁咖啡、美式咖啡)定制某些步骤(如冲泡咖啡和添加配料),这样做不仅减少了重复代码,还提高了系统的灵活性和可维护性。

核心总结

程序员必知!模板方法模式的实战应用与案例分析 - 程序员古德

模板方法模式在日常开发过程中使用的非常多,它能够封装不变部分,扩展可变部分,提取子类中的公共代码,集中到父类中,避免了代码的重复,此外,它还能控制子类的扩展,父类对模板方法进行定义,子类只能对特定步骤进行实现,保证了算法结构的一致性。模板方法也存在很严重缺点,它可能导致每个不同的实现都需要一个子类,这会增加类的数量,并可能使系统更加复杂,当子类存在很多时,会导致系统庞大,不利于维护。

使用模板方法时一定要明确模板方法和具体方法的责任边界,以及它们之间的协作方式,同时,要合理设计模板方法的抽象层次,既要保证足够的通用性,又要避免过度抽象导致的复杂性。

关注我,每天学习互联网编程技术 - 程序员古德

标签:...,咖啡,必知,void,咖啡豆,程序员,制作,模板
From: https://blog.51cto.com/bytegood/9210646

相关文章

  • 武汉星起航:亚马逊2024年燃爆销售季!新手卖家必知的平台攻略
    随着2023年步入尾声,亚马逊卖家们纷纷准备着2024年的销售攻势。在新的一年,有哪些值得关注的亚马逊营销节日?如何利用这些黄金时刻迅速收割平台流量,形成高效的产品订单转化?以下是武汉星起航总结的一份详尽攻略,助你在亚马逊的销售季中脱颖而出。1.关注亚马逊特色节日亚马逊每年都会推......
  • 已确认:软考教材2024年改版!程序员 会改版吗?
    2024年软考教材已确认改版,改版的消息已经正式公布。其中,系统集成项目管理工程师和信息系统监理师新书1月出版!上个月,我们发布过改版信息:软考教材即将改版,软考哪些科目会改版?如果你想了解哪些科目还可能会改版,可以点击:24年软考教材改版,这些科目会改吗?根据查询中国国家版本馆数据:系统......
  • C++模板例子
    title:"C++模板例子"date:2023-11-02T01:05:25+08:00tags:["C++"]categories:[]draft:false#include<vector>#include<type_traits>usingnamespacestd;classAA{};classBB{};classTest{public:template<cl......
  • java使用 template模板ftl 含有图片的生成数据
    点击查看代码/***Base64编码.*/publicstaticStringbase64Encode(byte[]input){BASE64Encoderencoder=newBASE64Encoder();returnencoder.encode(input);}@OverridepublicvoidprintStudentRxdjb(StudentRxdj......
  • 24年软考教材改版,程序员会改吗?
    前几天我们举办了一场技术类的公开课,不少考生都非常关心软考教材的改版问题。24年 程序员的教材会改版么?软考各科目最新资料免费领取>>>软考免费刷题:点此进入>>>01历年教材改版趋势软考涉及科目众多,很多教材已过多次改版,但从公开的渠道我们很难找到相关教材改版的汇总数据。因此......
  • 网络安全等级保护等级测评方案模板
    ......
  • Python Flask html 模板的继承
    前言全局说明一、安装flask模块官方源:pip3installflask==2.3.2国内源:pip3installflask==2.3.2-ihttp://pypi.douban.com/simple/--trusted-hostpypi.douban.com以上二选一,哪个安装快用哪个flask安装时间2023-11更多国内源:https://www.cnblogs.com/wutou......
  • #yyds干货盘点# LeetCode程序员面试金典:有序数组中的单一元素
    题目给你一个仅由整数组成的有序数组,其中每个元素都会出现两次,唯有一个数只会出现一次。请你找出并返回只出现一次的那个数。你设计的解决方案必须满足O(logn)时间复杂度和O(1)空间复杂度。 示例1:输入:nums=[1,1,2,3,3,4,4,8,8]输出:2示例2:输入:nums= [......
  • #yyds干货盘点# LeetCode程序员面试金典:UTF-8 编码验证
    题目给定一个表示数据的整数数组data,返回它是否为有效的UTF-8编码。UTF-8中的一个字符可能的长度为1到4字节,遵循以下的规则:对于1字节的字符,字节的第一位设为0,后面7位为这个符号的unicode码。对于n字节的字符(n>1),第一个字节的前n位都设为1,第n+1位......
  • 【设计模式】模板方法模式——模板方法模式在JDK源码中的应用
    模板方法模式在JDK源码里最典型的实现就是AQSAbstractQueuedSynchronizerAQS(AbstractQueuedSynchronizer)的部分代码如下:publicabstractclassAbstractQueuedSynchronizerextendsAbstractOwnableSynchronizerimplementsjava.io.Serializable{//……代码......