设计模式:模板模式(C#、JAVA、JavaScript、C++、Python、Go、PHP)

简介:

模板模式,它是一种行为型设计模式,它定义了一个操作中的算法的框架,将一些步骤延迟到子类中实现,使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
通俗地说,模板模式就是将某一行为制定一个框架,然后子类填充细节。比如说做菜,流程通常就是洗菜、切菜、炒菜等步骤,那么这个流程就可以看作是一个模板,而具体做什么菜由子类来实现。

模板模式的使用场景:
1、一次性实现一个算法的不变部分,并将可变的行为留给子类来实现。这种情况下,可以将一些通用的算法逻辑放在一个抽象类中,然后通过子类来提供具体的实现。
2、各子类中公共的行为被提取出来并集中到一个公共的父类中,从而避免代码重复。这种情况下,可以将一些共同的行为放在一个父类中,然后通过继承和多态来实现相似的功能,避免在多个子类中重复编写相同的代码。

例如,在网络销售商品中,模板流程大体为:上架商品-下订单-发快递-收货,如果需要退货还需进行退货流程。这种情况下,可以将发快递和退货等公共流程提取出来放到一个父类中,然后通过子类来实现具体的流程。

模板模式的创建步骤如下:
1、定义抽象类:定义一个抽象类,将要执行的操作封装在方法中。这些方法应该是一个模板方法,包含一些基本的操作步骤。
2、定义模板方法:在抽象类中定义一个模板方法,该方法指定了算法的骨架,并调用其他方法实现具体的操作步骤。模板方法通常是一些基本的流程控制结构,如循环、条件语句等。
3、实现抽象方法:在抽象类中定义一些抽象方法,这些方法将在子类中被实现。这些方法通常是一些具体的操作步骤,如数据处理、界面渲染等。
4、创建子类:创建具体的子类,并实现抽象类中的抽象方法。这些方法将根据实际需求进行定制。
5、实例化对象:在客户端代码中实例化一个对象,并调用模板方法开始执行算法。
这些步骤是模板方法模式的核心。在实际应用中,可以根据具体的需求进行修改和扩展。例如,可以添加更多的抽象方法来细化算法的步骤,或者使用不同的继承结构来组织代码。

模板模式的优点,主要包括:
1、封装不变部分:通过将不变的行为放在父类中,可以减少子类的重复代码,并将这些不变的行为在多个子类中复用。
2、提取公共代码:模板模式可以在多个子类中复用公共代码,使得代码更加简洁、易于维护和修改。
3、行为由父类控制,子类实现:模板模式使得行为的变化更加容易适应,因为行为的变化可以在父类中定义和控制,而子类只需要实现具体的行为即可。
4、提高可复用性:模板模式使得算法的框架可以在多个应用中使用,提高了代码的可复用性。
5、提高可维护性:模板模式使得代码的结构更加清晰,易于维护和理解。

模板模式的缺点,主要包括:
1、类的数量增加:为了实现模板方法,需要在父类中定义一个抽象方法,并且每个子类都需要实现该方法,这样就导致类的数量增加,从而增加了系统的复杂度。
2、继承的固有缺点:如果父类添加了新的抽象方法,所有子类都需要增加该方法,这就增加了代码的维护难度。另外,如果一个子类需要同时实现其他接口或继承其他类,就需要编写更多的代码。
3、实现难度增加:模板方法模式需要开发者自行设计算法框架和流程控制,如果算法较为复杂,实现难度就会增加。
4、不够灵活:模板方法模式相对固定,不够灵活,不能够很好地适应变化。如果需要修改算法框架,就需要修改所有子类的实现,工作量较大。


示例:

一、C#模板模式

在C#中,我们可以使用抽象类或接口来实现模板模式。以下是一个使用抽象类的示例:

public abstract class AbstractTemplate  
{  public void TemplateMethod()  {  Console.WriteLine("Subclass must implement steps");  }  public abstract void SpecificStep();  
}  public class ConcreteTemplate : AbstractTemplate  
{  public override void SpecificStep()  {  Console.WriteLine("This is a specific step");  }  
}
//可以这样使用模板模式:
public static void Main(string[] args)  
{  AbstractTemplate template = new ConcreteTemplate();  template.TemplateMethod(); // This will call SpecificStep and print "This is a specific step"  
}


二、java模板模式

模板模式通常通过以下方式实现:

public abstract class AbstractTemplate {  public void templateMethod() {  System.out.println("AbstractTemplate says:");  doSomething();  doSomethingElse();  }  public abstract void doSomething();  public abstract void doSomethingElse();  
}  public class ConcreteTemplate extends AbstractTemplate {  public void doSomething() {  System.out.println("ConcreteTemplate says:");  // specific behavior  }  public void doSomethingElse() {  System.out.println("ConcreteTemplate says:");  // specific behavior  }  
}
public class Client{public static void Main(String[] args){AbstractTemplate template = new ConcreteTemplate();template.templateMethod();	}
}

三、javascript模板模式

在JavaScript中,模板模式的实现方式如下:

function Person(name, age) {  this.name = name;  this.age = age;  
}  Person.prototype.sayHello = function() {  return `Hello, my name is ${this.name} and I am ${this.age} years old.`;  
};  const person1 = new Person("Alice", 25);  
const person2 = new Person("Bob", 30);  console.log(person1.sayHello()); // 输出:Hello, my name is Alice and I am 25 years old.  
console.log(person2.sayHello()); // 输出:Hello, my name is Bob and I am 30 years old.

四、C++模板模式

C++模板可以分为函数模板和类模板。函数模板定义了一个通用的函数框架,可以接受不同的参数类型,并生成相应的函数实现。类模板则定义了一个通用的类,可以接受不同的数据类型作为成员变量类型,并生成相应的类实现。

下面是一个使用C++函数模板的示例:

template <typename T>  
T add(T a, T b) {  return a + b;  
}  int main() {  int sum = add<int>(1, 2); // 使用整数类型参数  double avg = add<double>(1.0, 2.0); // 使用浮点数类型参数  return 0;  
}

在上面的示例中,函数模板 add 可以接受任意类型的参数 a 和 b,并返回它们的和。在 main 函数中,我们使用整数类型和浮点数类型分别调用 add 函数,生成相应的代码实现。

除了函数模板,C++还支持类模板。类模板可以定义一个通用的类,可以接受不同的数据类型作为成员变量类型,并生成相应的类实现。下面是一个使用C++类模板的示例:
 

template <typename T>  
class Array {  
public:  Array(int size) : size_(size), data_(new T[size]) {}  ~Array() { delete[] data_; }  T& operator[](int index) { return data_[index]; }  
private:  int size_;  T* data_;  
};  int main() {  Array<int> int_array(10); // 使用整数类型参数  Array<double> double_array(5); // 使用浮点数类型参数  return 0;  
}

在上面的示例中,类模板 Array 可以接受任意类型的参数 T 作为成员变量类型,并生成相应的类实现。在 main 函数中,我们使用整数类型和浮点数类型分别创建 Array 类的对象,生成相应的代码实现。

五、python模板模式

在python中有两种方式实现模板模式。
1、使用装饰器来扩展函数的行为来实现。装饰器可以在不修改函数代码的情况下添加新的行为。下面是一个简单的例子:

def template_method(func):  def wrapper(*args, **kwargs):  print("Before the function is called.")  result = func(*args, **kwargs)  print("After the function is called.")  return result  return wrapper  @template_method  
def add(x, y):  return x + y  print(add(2, 3))

在这个例子中,template_method 是一个装饰器,它定义了一个框架,在调用函数之前和之后打印一些信息。add 函数被装饰器装饰,因此它继承了装饰器的行为。当我们调用 add 函数时,它将在调用之前和之后打印信息,并返回函数的结果。

2、使用抽象基类(ABC)和多态来实现。抽象基类定义了一个通用的接口,子类必须实现这个接口。多态允许使用不同的子类来扩展父类的行为。下面是一个使用抽象基类和多态的例子:

from abc import ABC, abstractmethod  class Shape(ABC):  def draw(self):  pass  class Circle(Shape):  def draw(self):  print("Drawing circle")  class Rectangle(Shape):  def draw(self):  print("Drawing rectangle")  def draw_shape(shape):  shape.draw()  if __name__ == "__main__":  shapes = [Circle(), Rectangle()]  for shape in shapes:  draw_shape(shape)

在这个例子中,Shape 是一个抽象基类,它定义了一个 draw 方法,子类必须实现这个方法。Circle 和 Rectangle 是 Shape 的子类,它们分别实现了 draw 方法。draw_shape 函数是一个模板方法,它接受一个 Shape 对象作为参数,并调用其 draw 方法。在主程序中,我们创建了一个包含不同形状的列表,并使用 draw_shape 函数来绘制它们。由于它们都继承自 Shape 类,因此它们都可以被 draw_shape 函数接受并正确绘制。

六、go模板模式

在Go语言中,没有直接称为"模板模式"的设计模式,但是有一种类似的行为可以通过使用函数和闭包来实现。
在Go语言中,可以通过定义一个父类和一个子类来实现类似模板方法模式的行为。父类可以包含一个或多个函数,这些函数可以调用子类的方法来实现算法的某些步骤。子类可以继承父类的函数,并重写其中的一些方法来改变算法的行为。

下面是一个简单的示例代码,实现类似模板方法模式的行为:

package main  import "fmt"  type Parent struct{}  func (p *Parent) Algorithm() {  fmt.Println("Parent's Algorithm")  p.Step1()  p.Step2()  
}  func (p *Parent) Step1() {  fmt.Println("Parent's Step 1")  
}  func (p *Parent) Step2() {  fmt.Println("Parent's Step 2")  
}  type Child struct {  Parent  
}  func (c *Child) Step1() {  fmt.Println("Child's Step 1")  
}  func main() {  child := &Child{}  child.Algorithm() // Output: Parent's Algorithm, Child's Step 1, Parent's Step 2  
}

七、PHP模板模式

在PHP中,模板模式通常使用模板引擎来实现。

模板引擎是一种独立的中间件,用于处理模板和生成最终的输出。它负责解析模板文件,将动态数据与模板文件中的占位符进行替换,并将最终的输出发送给客户端。PHP中有很多流行的模板引擎,如Smarty、Twig和Blade等。

下面是一个简单的示例,演示了如何使用PHP模板模式:

//创建模板文件(template.php):
<html>  
<head>  <title>{{ title }}</title>  
</head>  
<body>  <h1>{{ message }}</h1>  
</body>  
</html>
//创建PHP脚本(script.php):
<?php  
// 加载并配置模板引擎(以Twig为例)  
require_once 'vendor/autoload.php'; // 根据实际情况加载Composer依赖项  
$loader = new \Twig\Loader\FilesystemLoader('path/to/templates');  
$twig = new \Twig\Environment($loader);  // 准备动态数据  
$data = [  'title' => 'Welcome',  'message' => 'Hello, world!'  
];  // 渲染模板并输出结果  
echo $twig->render('template.php', $data);

在上面的示例中,我们使用了Twig模板引擎来解析和渲染模板文件。在script.php中,我们首先加载并配置了Twig引擎,然后准备了一个包含动态数据的数组。最后,我们使用Twig的render方法将动态数据与模板文件进行替换,并将最终的输出发送给客户端。

《完结》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/145461.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【ARM裸机】ARM入门

1.ARM成长史 2.ARM的商业模式和生态系统 ARM只设计CPU&#xff0c;但是不生产CPU 3.为什么使用三星&#xff1a;S5PV210 4.各种版本号 0. ARM和Cortex Cortex就是ARM公司一个系列处理器的名称。比如英特尔旗下处理器有酷睿&#xff0c;奔腾&#xff0c;赛扬。ARM在最初的处理器…

程序包org.apache.ibatis.mapping不存在 符号找不到

找不到符号 符号: 类 Cursor和程序包org.apache.ibatis.mapping不存在 在idea中没有错误&#xff0c;但是在linux编辑时报了这两个错误&#xff0c;之前有遇见过符号找不到的问题&#xff0c; 当时的问题是编译的import xxx.xxx.xxx.* 识别不成功过&#xff0c;将*改为…

网络协议--ICMP:Internet控制报文协议

6.1 引言 ICMP经常被认为是IP层的一个组成部分。它传递差错报文以及其他需要注意的信息。ICMP报文通常被IP层或更高层协议&#xff08;TCP或UDP&#xff09;使用。一些ICMP报文把差错报文返回给用户进程。 ICMP报文是在IP数据报内部被传输的&#xff0c;如图6-1所示。 ICMP…

卷积神经网络CNN学习笔记-MaxPool2D函数解析

目录 1.函数签名:2.学习中的疑问3.代码 1.函数签名: torch.nn.MaxPool2d(kernel_size, strideNone, padding0, dilation1, return_indicesFalse, ceil_modeFalse) 2.学习中的疑问 Q:使用MaxPool2D池化时,当卷积核移动到某位置,该卷积核覆盖区域超过了输入尺寸时,MaxPool2D会…

为什么学完了 C#觉得自己什么都干不了?

为什么学完了 C#觉得自己什么都干不了&#xff1f; 俺一向的观点&#xff1a;&#xff08;1&#xff09;学跟干是两码事。学&#xff0c;你要往外掏钱或时间或两个都得掏。干&#xff0c;是你从别人兜里掏钱。&#xff08;2&#xff09;如果没有干的需求&#xff0c;那么可以啥…

小白也会的校园网宽带拨号自动重连设置

开始菜单搜索“任务计划程序”&#xff1a;我这个开始菜单和你们的不太一样&#xff0c;用了StartAllBack设置的&#xff0c;总之能找到这个程序就行了 提示&#xff1a;可以按下“Win R”&#xff0c;打开“运行”&#xff0c;输入taskschd.msc来打开任务计划程序 点击“任务…

灰度发布专题---1、灰度发布的意义和方案

线上项目灰度发布的重大意义 支付宝等软件相信大家一定不陌生&#xff0c;支付宝经历了十多年&#xff0c;从未停止更新过&#xff0c;app从最初简单设计到现在的扁平化设计&#xff0c;一直在更新&#xff0c;但奇怪的是它从未停过服务&#xff0c;而且越用越顺畅。不停服务就…

Java程序设计2023-第四次上机练习

8-1三子棋 编写程序&#xff0c;实现简单的三子棋游戏。在三子棋中&#xff0c;双方在33的棋盘中轮流下棋&#xff0c;一方用*示&#xff0c;另一方用O表示。如果一方的3个棋子占据了同一行&#xff0c;同一列或者对角线&#xff0c;则该方获胜。如果棋盘已被棋子占满&#xf…

【Mongo】数据删了磁盘空间但没有减少

Author:skate Time:2023/10/22 一、问题描述 产线用户反馈&#xff0c;一个华为云的mongo实例磁盘空间告警&#xff0c;使用率超过90%&#xff08;使用状况 1630.9/1800GB&#xff09;&#xff0c;让其通过数据库运维平台找到占用大空间的表&#xff0c;然后清理历史数据&…

Centos磁盘问题小纪

场景说明 放个windows的图片镇楼&#xff0c;在给一个centos的来说明问题&#xff0c;咋了&#xff0c;好好的系统&#xff0c;啥也不能干了 来先上一波命令分析下问题 查看挂载 mount 重新挂载数据 mount -o remount, rw / 查看磁盘 df -h 查看分区挂载详情 rw读写权限 mount …

react封装一个简单的upload组件(待完善)

目录 react封装一个简单的upload组件component / uploadImg / uploadImg.jsx使用效果 react封装一个简单的upload组件 component / uploadImg / uploadImg.jsx import React, { useState } from react; import { LoadingOutlined, PlusOutlined } from ant-design/icons; imp…

【leetcode报错】 leetcode格式问题解决:error: stray ‘\302’ in program [solution.c]

leetcode格式问题解决 一、情景再现二、报错原因三、解决方法四、修正结果 一、情景再现 二、报错原因 该错误是指 源程序中有非法字符&#xff0c;需要将非法字符去掉。 一般是由于coder 1.使用中文输入法 或者 2.从别的地方直接复制粘贴代码 造成的。 代码中出现了 中文空格&…