go语言设计模式<一>模板方法

适合流程固定,有些步骤有特殊实现的情况。类图如下。优缺点不分析了,直接上实现

go 复制代码
package main

import (
	"fmt"
)

// Template 接口
type Template interface {
	TemplateMethod(template Template)
	// 钩子方法,需要子类去具体的实现
	step1Func()
	exposedFunc()
	step3Func()
}

// BaseTemplate 基础模板,编排方法,提供复用的方法
type BaseTemplate struct {
}

// NewBaseTemplate 创建基础模板
func NewBaseTemplate() *BaseTemplate {
	return &BaseTemplate{}
}

// TemplateMethod 模板方法
func (b *BaseTemplate) TemplateMethod(template Template) {
	fmt.Println("=== 开始执行模板方法 ===")
	if template == nil {
		return
	}
	template.step1Func()
	template.exposedFunc()
	template.step3Func()
	fmt.Println("=== 模板方法执行完成 ===")
}

// 基础方法1:子类可以不实现
func (b *BaseTemplate) step1Func() {
	fmt.Println("=== base.step1Func is called ===")
}

// 钩子方法,子类必须实现
func (b *BaseTemplate) exposedFunc() {
	panic("exposed Func must be implement!")
}

// 基础方法2:子类可以直接继承基类的,可以不实现
func (b *BaseTemplate) step3Func() {
	fmt.Println("=== base.step3Func is called ===")
}

// ==================== 具体实现 ====================

// ConcreteTemplateA
type ConcreteTemplateA struct {
	*BaseTemplate
}

func NewConcreteTemplateA() *ConcreteTemplateA {
	base := NewBaseTemplate()
	c := &ConcreteTemplateA{BaseTemplate: base}

	return c
}

func (c *ConcreteTemplateA) exposedFunc() {
	fmt.Println("具体模板A: exposedFunc - 特殊实现")
}

// ConcreteTemplateB
type ConcreteTemplateB struct {
	*BaseTemplate
}

func NewConcreteTemplateB() *ConcreteTemplateB {
	base := NewBaseTemplate()
	c := &ConcreteTemplateB{BaseTemplate: base}
	return c
}

func (c *ConcreteTemplateB) step1Func() {
	fmt.Println("具体模板B:step1Func override")
}

func (c *ConcreteTemplateB) exposedFunc() {
	fmt.Println("具体模板B: exposedFunc -- 钩子函数实现")
}

// ==================== 使用示例 ====================

func InitTemplate(templateKey string) Template {
	// 路由判断使用哪套模板
	mp := make(map[string]Template)
	mp["templateA"] = NewConcreteTemplateA()
	mp["templateB"] = NewConcreteTemplateA()

	if ans, exist := mp[templateKey]; exist {
		return ans
	}
	panic(fmt.Sprintf("There is no template named %s", templateKey))
}

func main() {
	templateA := InitTemplate("templateA")
	templateB := InitTemplate("templateB")
	var baseTemplate Template = NewBaseTemplate() // 可以不要

	fmt.Println("=== 模板A执行 ===")
	baseTemplate.TemplateMethod(templateA)

	fmt.Println("\n=== 模板B执行 ===")
	baseTemplate.TemplateMethod(templateB)
}
相关推荐
lead520lyq几秒前
Golang Grpc接口调用实现账号密码认证
开发语言·后端·golang
EQ-雪梨蛋花汤几秒前
【问题反馈】JNI 开发:为什么 C++ 在 Debug 正常,Release 却返回 NaN?
开发语言·c++
naruto_lnq1 分钟前
高性能消息队列实现
开发语言·c++·算法
charlie1145141911 分钟前
malloc 在多线程下为什么慢?——从原理到实测
开发语言·c++·笔记·学习·工程实践
kyrie学java4 分钟前
SpringWeb
java·开发语言
写代码的【黑咖啡】6 分钟前
Python 中的 Gensim 库详解
开发语言·python
小二·7 分钟前
Go 语言系统编程与云原生开发实战(第6篇)云原生部署实战:Docker 镜像瘦身 × K8s 部署 × Helm 一键发布
docker·云原生·golang
饺子大魔王的男人2 小时前
Remote JVM Debug+cpolar 让 Java 远程调试超丝滑
java·开发语言·jvm
花酒锄作田8 小时前
MCP官方Go SDK尝鲜
golang·mcp
兩尛8 小时前
c++知识点2
开发语言·c++