模板方法设计模式是一种行为型设计模式。这种模式通过常用于为某种特定的操作定义一个模板或者算法模型。

以一次性密码(OTP:One Time Password)为例。我们常见的一次性密码有两种:短信密码(SMS OTP)或者邮件密码(Email OTP)。不过不管是短信密码还是邮件密码,它们的处理步骤都是一样的,步骤如下:

  1. 生成一串随机字符串
  2. 将字符串保存进缓存用来执行后续的验证
  3. 准备通知内容
  4. 发送通知
  5. 记录统计信息

在以上的步骤中,除了第4项“发送通知”的具体方式不一样,其他步骤都是不变的。即使以后有了新的一次性密码发送方式,可以预见以上的步骤也是不变的。

在这类场景中,即一个操作的步骤是固定的,只是在具体的执行方式上存在差异,这时我们就可以用到模板方法模式了。在模板方法模式中,我们通常会为这个操作定义一个模板接口或算法模型接口,接口中包含固定方法,然后由具体的实现类来重写相关接口并实现这些操作。

下面是一次性密码这个例子的实现:

otp.go

  1. type iOtp interface {
  2. genRandomOTP(int) string
  3. saveOTPCache(string)
  4. getMessage(string) string
  5. sendNotification(string) error
  6. publishMetric()
  7. }
  8.  
  9. type otp struct {
  10. iOtp iOtp
  11. }
  12.  
  13. func (o *otp) genAndSendOTP(otpLength int) error {
  14. otp := o.iOtp.genRandomOTP(otpLength)
  15. o.iOtp.saveOTPCache(otp)
  16. message := o.iOtp.getMessage(otp)
  17. err := o.iOtp.sendNotification(message)
  18. if err != nil {
  19. return err
  20. }
  21. o.iOtp.publishMetric()
  22. return nil
  23. }

简单解读下这段代码:

  • 在上面的代码中定义了iOtp接口,这个接口中的方法即OTP所需的相关步骤
  • 下面的smsemailiOtp接口的实现
  • 在结构体otp中定义了模板方法genAndSendOTP()

此外注意下:在上面这段代码中将iOtp接口和结构体otp合在了一起,提供了类似于抽象类的实现,这种做法大家需要的时候可以借鉴下。

sms.go

  1. import "fmt"
  2.  
  3. type sms struct {
  4. otp
  5. }
  6.  
  7. func (s *sms) genRandomOTP(len int) string {
  8. randomOTP := "1234"
  9. fmt.Printf("SMS: generating random otp %s\n", randomOTP)
  10. return randomOTP
  11. }
  12.  
  13. func (s *sms) saveOTPCache(otp string) {
  14. fmt.Printf("SMS: saving otp: %s to cache\n", otp)
  15. }
  16.  
  17. func (s *sms) getMessage(otp string) string {
  18. return "SMS OTP for login is " + otp
  19. }
  20.  
  21. func (s *sms) sendNotification(message string) error {
  22. fmt.Printf("SMS: sending sms: %s\n", message)
  23. return nil
  24. }
  25.  
  26. func (s *sms) publishMetric() {
  27. fmt.Printf("SMS: publishing metrics\n")
  28. }

email.go

  1. import "fmt"
  2.  
  3. type email struct {
  4. otp
  5. }
  6.  
  7. func (s *email) genRandomOTP(len int) string {
  8. randomOTP := "1234"
  9. fmt.Printf("EMAIL: generating random otp %s\n", randomOTP)
  10. return randomOTP
  11. }
  12.  
  13. func (s *email) saveOTPCache(otp string) {
  14. fmt.Printf("EMAIL: saving otp: %s to cache\n", otp)
  15. }
  16.  
  17. func (s *email) getMessage(otp string) string {
  18. return "EMAIL OTP for login is " + otp
  19. }
  20.  
  21. func (s *email) sendNotification(message string) error {
  22. fmt.Printf("EMAIL: sending email: %s\n", message)
  23. return nil
  24. }
  25.  
  26. func (s *email) publishMetric() {
  27. fmt.Printf("EMAIL: publishing metrics\n")
  28. }

main.go

  1. import "fmt"
  2.  
  3. func main() {
  4. smsOTP := &sms{}
  5. o := otp{
  6. iOtp: smsOTP,
  7. }
  8. o.genAndSendOTP(4)
  9. fmt.Println("")
  10. emailOTP := &email{}
  11. o = otp{
  12. iOtp: emailOTP,
  13. }
  14. o.genAndSendOTP(4)
  15. }

输出内容为:

  1. SMS: generating random otp 1234
  2. SMS: saving otp: 1234 to cache
  3. SMS: sending sms: SMS OTP for login is 1234
  4. SMS: publishing metrics
  5.  
  6. EMAIL: generating random otp 1234
  7. EMAIL: saving otp: 1234 to cache
  8. EMAIL: sending email: EMAIL OTP for login is 1234
  9. EMAIL: publishing metrics

代码已上传至GitHub: zhyea / go-patterns / template-method-pattern

END!!

GoLang设计模式16 - 模板方法模式的更多相关文章

  1. 乐在其中设计模式(C#) - 模板方法模式(Template Method Pattern)

    原文:乐在其中设计模式(C#) - 模板方法模式(Template Method Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 模板方法模式(Template Method ...

  2. 折腾Java设计模式之模板方法模式

    博客原文地址:折腾Java设计模式之模板方法模式 模板方法模式 Define the skeleton of an algorithm in an operation, deferring some ...

  3. Golang设计模式—简单工厂模式(Simple Factory Pattern)

    Golang设计模式--简单工厂模式 背景 假设我们在做一款小型翻译软件,软件可以将德语.英语.日语都翻译成目标中文,并显示在前端. 思路 我们会有三个具体的语言翻译结构体,或许以后还有更多,但现在分 ...

  4. js设计模式——6.模板方法模式与职责链模式

    js设计模式——6.模板方法模式与职责链模式 职责链模式

  5. java设计模式之模板方法模式

    模板方法模式 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中. 模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤.通俗的说的就是有很多相同的步骤的,在某一些地方可能有一些差 ...

  6. C#设计模式(14)——模板方法模式(Template Method)

    一.引言 提到模板,大家肯定不免想到生活中的“简历模板”.“论文模板”.“Word中模版文件”等,在现实生活中,模板的概念就是——有一个规定的格式,然后每个人都可以根据自己的需求或情况去更新它,例如简 ...

  7. 【GOF23设计模式】模板方法模式

    来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_模板方法模式.钩子函数.方法回调.好莱坞原则 package com.test.templateMethod; publi ...

  8. [设计模式] 22 模板方法模式 template

    转http://www.jellythink.com/archives/407 在GOF的<设计模式:可复用面向对象软件的基础>一书中对模板方法模式是这样说的:定义一个操作中的算法骨架,而 ...

  9. java_设计模式_模板方法模式_Template Method Pattern(2016-08-11)

    定义: 定义一个操作中算法的骨架,而将一些步骤延迟到子类中,使得子类可以不改变算法的结构即可重定义该算法中的某些特定步骤.这里的算法的结构,可以理解为你根据需求设计出来的业务流程.特定的步骤就是指那些 ...

随机推荐

  1. 洛谷3195 [HNOI2008]玩具装箱TOY(斜率优化+dp)

    qwq斜率优化好题 第一步还是考虑最朴素的\(dp\) \[dp=dp[j]+(i-j-1+sum[i]-sum[j])^2 \] 设\(f[i]=sum[i]+i\) 那么考虑将上述柿子变成$$dp ...

  2. 从0到1使用Kubernetes系列(三):使用Ansible安装Kubernetes集群

    前两期的文章介绍了Kubernetes基本概念和架构,用Kubeadm+Ansible搭建Kubernetes集群所需要的工具及其作用.本篇介绍怎么使用Ansible安装Kubernetes集群. 启 ...

  3. Endian

    Endian 寻址 多字节对象被存储为连续的字节序列,对象的地址为所使用字节中最小的地址. 例如,假设一个类型为 int 的变量 a 的地址为 0x100,也就是说,地址表达式 &a 的值为 ...

  4. java定时任务调度框架

    java定时任务目前主要有三种: Java自带的java.util.Timer类,这个类允许你调度一个java.util.TimerTask任务.使用这种方式可以让你的程序按照某一个频度执行,但不能在 ...

  5. 整数划分为k份

    题目 将整数n分成k份,且每份不能为空,任意两个方案不能相同(不考虑顺序). 例如:n=7,k=3,下面三种分法被认为是相同的. 1,1,5; 1,5,1; 5,1,1; 问有多少种不同的分法. 输入 ...

  6. Noip模拟19(炸裂的开始) 2021.7.18

    T1 u 差分与前缀的综合练习. 分析数据范围,只能是在修改的时候$O(1)$做到,那么只能是像打标记一样处理那个三角形 正解是建立两个二位前缀和,一个控制竖向,一个控制斜向 每次在三角的左上,右下, ...

  7. noip模拟11

    T1 math 就挺水一小破题目,第一眼看好像不可做,看着看着突然发现假设x和y的最大公约数是gcd,那么kx%y一定是gcd的倍数, 然后想到可以把所有数字与k的gcd求出来,打一个完全背包,可是仔 ...

  8. 字符串与模式匹配算法(六):Needleman–Wunsch算法

    一.Needleman-Wunsch 算法 尼德曼-翁施算法(英语:Needleman-Wunsch Algorithm)是基于生物信息学的知识来匹配蛋白序列或者DNA序列的算法.这是将动态算法应用于 ...

  9. vcs命令

    转载:VCS_weixin_34256074的博客-CSDN博客 timing check相关的: +notimingcheck命令,可以用在compile时,也可以用在run time的时候, 都是 ...

  10. poj 2960 S-Nim (SG)

    题意: K个数,s1...sk. m个状态,对于某一个状态,有L堆石子,每人每次取的石子个数只能是s1...sk的一个,且只能在一堆中取. 输出m个状态是先手胜还是先手败,先手胜输出W,否则输出L. ...