c#设计模式-行为型模式 之 模板方法模式

🚀简介

        模板方法模式定义了一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。通常用于应对在开发中设计一个系统时知道了算法所需的关键步骤,而且确定了这些步骤的执行顺序,但某些步骤的具体实现还未知,或者说某些步骤的实现与具体的环境相关。

👻角色

模板方法( Template Method )模式包含以下主要角色:
  1. 抽象类(Abstract Class):负责给出一个算法的轮廓和骨架。它由一个模板方法和若干个基 本方法构成。
    1. 模板方法:定义了算法的骨架,按某种顺序调用其包含的基本方法。
    2. 基本方法:是实现算法各个步骤的方法,是模板方法的组成部分。基本方法又可以分为三 种:
      1. 抽象方法(Abstract Method) :一个抽象方法由抽象类声明、由其具体子类实现。
      2. 具体方法(Concrete Method) :一个具体方法由一个抽象类或具体类声明并实现, 其子类可以进行覆盖也可以直接继承。
      3. 钩子方法(Hook Method) :在抽象类中已经实现,包括用于判断的逻辑方法和需要子类重写的空方法两种。一般钩子方法是用于判断的逻辑方法,这类方法名一般为isXxx,返回值类型为boolean类型。
  2. 具体子类(Concrete Class):实现抽象类中所定义的抽象方法和钩子方法,它们是一个顶级逻辑的组成步骤

🚀案例

例如我们在炒菜的时候,炒菜的步骤是固定的,分为倒油、热油、倒菜、倒调料品、翻炒等步骤。现通过模板方法模式来用代码模拟

🐤抽象类

在以下抽象类中,有具体的实现方法,如倒油、热油、翻炒,因为这些步骤是我们已知的,不管炒什么菜都需要经历这个步骤,但是炒的菜不一样我们倒的蔬菜就不一样,加的调料也不一样,因此把这一部分声明为抽象方法,在子类中再去实现。最后在cookProcess中按照顺序调用类中的方法。

public abstract class AbstractClass
{
    //第一步倒油,直接实现
    public void pourOil()
    {
        Console.WriteLine("倒油");
    }

    //第二步:热油是一样的,直接实现
    public void heatOil()
    {
        Console.WriteLine("热油");
    }

    //第三步:倒蔬菜
    public abstract void pourVegetable();

    //第四步:倒调味料是不一样
    public abstract void pourSauce();

    //第五步:翻炒是一样的,所以直接实现
    public void fry()
    {
        Console.WriteLine("炒啊炒啊炒到熟啊");
    }

    public void cookProcess()
    {
        //第一步:倒油
        this.pourOil();
        //第二步:热油
        this.heatOil();
        //第三步:倒蔬菜
        this.pourVegetable();
        //第四步:倒调味料
        this.pourSauce();
        //第五步:翻炒
        this.fry();
    }
}

🐤具体子类

炒辣包菜,实现抽象类中关于菜和配料的抽象方法。

public class ConcreteClass_BaoCai : AbstractClass
{
    public override void pourVegetable()
    {
        Console.WriteLine("下包菜");
    }

    public override void pourSauce()
    {
        Console.WriteLine("下辣椒");
    }
}

🐳同上

public class ConcreteClass_CaiXin : AbstractClass
{
    public override void pourVegetable()
    {
        Console.WriteLine("下菜心");
    }

    public override void pourSauce()
    {
        Console.WriteLine("下蒜蓉");
    }
}

🐤测试类

class MyClass
{
    public static void Main(string[] args)
    {
        var concreteClassBaoCai = new ConcreteClass_BaoCai();
        concreteClassBaoCai.cookProcess();
        Console.WriteLine("---------------------");
        var concreteClassCaiXin = new ConcreteClass_CaiXin();
        concreteClassCaiXin.cookProcess();
    }
}

运行结果!

🚀优缺点

优点:
  1. 提高代码复用性,将相同部分的代码放在抽象的父类中,而将不同的代码放入不同的子类中。
  2. 实现了反向控制,通过一个父类调用其子类的操作,通过对子类的具体实现扩展不同的行为,实现了反向控制 ,并符合开闭原则
缺点:
  1. 对每个不同的实现都需要定义一个子类,这会导致类的个数增加,系统更加庞大,设计也更加抽象。
  2. 父类中的抽象方法由子类实现,子类执行的结果会影响父类的结果,这导致一种反向的控制结构,它提高了代码阅读的难度。
适用场景
  1. 算法的整体步骤很固定,但其中个别部分易变时,这时候可以使用模板方法模式,将容易变的部分抽象出来,供子类实现。
  2. 需要通过子类来决定父类算法中某个步骤是否执行,实现子类对父类的反向控制。

http://www.niftyadmin.cn/n/5065438.html

相关文章

【Java-LangChain:使用 ChatGPT API 搭建系统-4】评估输入-分类

第三章,评估输入-分类 如果您正在构建一个允许用户输入信息的系统,首先要确保人们在负责任地使用系统,以及他们没有试图以某种方式滥用系统,这是非常重要的。 在本章中,我们将介绍几种策略来实现这一目标。 我们将学习…

41 二叉树的层序遍历

二叉树的层序遍历 题解1 迭代——BFS题解2 递归——DFS 给你二叉树的根节点 root &#xff0c;返回其节点值的 层序遍历。 &#xff08;即逐层地&#xff0c;从左到右访问所有节点&#xff09;。 提示&#xff1a; 树中节点数目在范围 [0, 2000] 内-1000 < Node.val <…

MATLAB算法实战应用案例精讲-【优化算法】A*算法

前言 A*算法最早于1964年在IEEE Transactions on Systems Science and Cybernetics中的论文《A Formal Basis for the Heuristic Determination of Minimum Cost Paths》中首次提出。 其属于一种经典的启发式搜索方法,所谓启发式搜索,就在于当前搜索结点往下选择下一步结点时…

go语法入门1

语法基础 注释 // 单行注释 /* xxxx */ 编译器忽略该区间&#xff0c;其间都被认为是注释内容。虽然Go支持&#xff0c;但很少使用&#xff0c;一般都用多行// //包注释 package mainimport "fmt"/* x int y int */// x int // y int 多行注释func main() {fmt.P…

链表--合并多个有序链表

例 输入&#xff1a;lists [[1,4,5],[1,3,4],[2,6]] 输出&#xff1a;[1,1,2,3,4,4,5,6] 解释&#xff1a;链表数组如下&#xff1a; [1->4->5,1->3->4,2->6 ] 将它们合并到一个有序链表中得到。 1->1->2->3->4->4->5->6 如图&#xff…

java Spring Boot 手动启动热部署

好 接下来 我们讲一个对开发非常重要的东西 热部署 因为 我们在开发过程中总会希望快点看到效果 或者 你的企业项目一般很大很复杂&#xff0c;重启是一件非常麻烦的事 或者你在和前端同事联调&#xff0c;有一点小问题 你改完就要重启 前端还得等你&#xff0c;非常不友好 那…

二十八、高级IO与多路转接之select

文章目录 一、五种IO模型&#xff08;一&#xff09;阻塞IO:&#xff08;二&#xff09;非阻塞IO:&#xff08;三&#xff09;信号驱动IO:&#xff08;四&#xff09;IO多路转接:&#xff08;五&#xff09;异步IO: 二、高级IO重要概念&#xff08;一&#xff09;同步通信 vs 异…

TCP端口崩溃,msg:socket(): Too many open files

一、现象 linux系统中运行了一个TCP服务器&#xff0c;该服务器监听的TCP端口为10000。但是长时间运行时发现该端口会崩溃&#xff0c;TCP客户端连接该端口会失败&#xff1a; 可以看到进行三次握手时&#xff0c;TCP客户端向该TCP服务器的10000端口发送了SYN报文&#xff0c;…