Go可以使用设计模式,但绝不是《设计模式》中的那样

文章精选推荐

1 JetBrains Ai assistant 编程工具让你的工作效率翻倍
2 Extra Icons:JetBrains IDE的图标增强神器
3 IDEA插件推荐-SequenceDiagram,自动生成时序图
4 BashSupport Pro 这个ides插件主要是用来干嘛的 ?
5 IDEA必装的插件:Spring Boot Helper的使用与功能特点
6 Ai assistant ,又是一个写代码神器
7 Cursor 设备ID修改器,你的Cursor又可以继续试用了

文章正文

Go语言是一门简洁、高效的编程语言,它支持面向对象编程的一些特性,比如接口、封装和组合,但并不直接提供类、继承等传统的面向对象概念。

因此,Go语言在使用设计模式时,有一些独特的方式和哲学,与传统面向对象语言(如 Java、C++)中的设计模式实现方式有所不同。

本文将深入探讨 Go 如何使用设计模式,并强调与《设计模式》中的传统实现的不同点,并通过代码示例进行说明。


1. Go 中的设计哲学

在 Go 中,有以下设计哲学与传统设计模式实现密切相关:

  • 组合优于继承:Go 提倡通过组合而不是继承来实现代码复用,避免了复杂的类层次结构。
  • 接口解耦:接口是 Go 语言中非常核心的特性,允许我们定义行为契约,而不依赖具体实现。
  • 简洁优先:Go 鼓励通过简单、直接的方式解决问题,而不是引入复杂的模式。

因此,在 Go 中,很多经典设计模式会简化,甚至在某些情况下,Go 原生特性(如接口、goroutine)已经能直接解决问题。


2. Go 中的常见设计模式解析

2.1 单例模式

传统实现
在面向对象语言中,单例模式通常需要通过私有构造函数、静态变量和双重检查锁定来实现线程安全的单例。

Go 实现
在 Go 中,可以通过 sync.Once 来实现线程安全的单例模式,代码更加简洁。

package mainimport ("fmt""sync"
)// 单例结构体
type Singleton struct{}var (instance *Singletononce     sync.Once
)// 获取单例实例
func GetInstance() *Singleton {once.Do(func() {instance = &Singleton{}})return instance
}func main() {s1 := GetInstance()s2 := GetInstance()fmt.Println(s1 == s2) // true
}

关键点

  • 使用 sync.Once 确保单例只初始化一次,避免手动实现锁和双重检查。

2.2 工厂模式

传统实现
在传统语言中,工厂模式通常通过一个基类(或接口)和具体子类实现。

Go 实现
由于 Go 语言没有类,工厂模式可以通过接口来抽象行为,通过函数直接创建实例。

package mainimport "fmt"// 动物接口
type Animal interface {Speak() string
}// 狗的实现
type Dog struct{}func (d Dog) Speak() string {return "Woof!"
}// 猫的实现
type Cat struct{}func (c Cat) Speak() string {return "Meow!"
}// 工厂函数
func NewAnimal(animalType string) Animal {switch animalType {case "dog":return Dog{}case "cat":return Cat{}default:return nil}
}func main() {dog := NewAnimal("dog")fmt.Println(dog.Speak()) // Woof!cat := NewAnimal("cat")fmt.Println(cat.Speak()) // Meow!
}

关键点

  • 使用简单的工厂函数代替复杂的类和构造函数。
  • 动物的行为通过接口抽象,而具体类型由工厂函数决定。

2.3 策略模式

传统实现
策略模式通常通过抽象基类和多个具体子类来实现不同策略。

Go 实现
在 Go 中,可以直接使用接口和函数作为策略实现。

package mainimport "fmt"// 策略接口
type PaymentStrategy interface {Pay(amount float64)
}// 信用卡支付
type CreditCard struct{}func (cc CreditCard) Pay(amount float64) {fmt.Printf("Paid %.2f using Credit Card.\n", amount)
}// PayPal 支付
type PayPal struct{}func (pp PayPal) Pay(amount float64) {fmt.Printf("Paid %.2f using PayPal.\n", amount)
}// 使用策略模式
func ProcessPayment(strategy PaymentStrategy, amount float64) {strategy.Pay(amount)
}func main() {cc := CreditCard{}pp := PayPal{}ProcessPayment(cc, 100.0) // Paid 100.00 using Credit Card.ProcessPayment(pp, 200.0) // Paid 200.00 using PayPal.
}

关键点

  • 使用接口定义策略行为,通过具体实现来提供不同的策略。
  • 策略选择可以在运行时动态更改。

2.4 观察者模式

传统实现
观察者模式通过一个主题(Subject)和多个观察者(Observer)实现,主题通知观察者更新。

Go 实现
在 Go 中,可以通过通道(channel)实现观察者模式,利用 goroutine 提供异步通知。

package mainimport "fmt"// 观察者
type Observer interface {Update(data string)
}// 具体观察者
type ConcreteObserver struct {id string
}func (co *ConcreteObserver) Update(data string) {fmt.Printf("Observer %s received: %s\n", co.id, data)
}// 主题
type Subject struct {observers []Observer
}func (s *Subject) Register(observer Observer) {s.observers = append(s.observers, observer)
}func (s *Subject) Notify(data string) {for _, observer := range s.observers {observer.Update(data)}
}func main() {subject := &Subject{}observer1 := &ConcreteObserver{id: "1"}observer2 := &ConcreteObserver{id: "2"}subject.Register(observer1)subject.Register(observer2)subject.Notify("Hello Observers!")
}

关键点

  • 使用接口定义观察者行为。
  • Notify 方法遍历所有观察者并调用其 Update 方法。

3. Go 的特色替代方案

在某些情况下,Go 的原生特性(如 goroutine、channel)可以替代传统设计模式。

3.1 使用 goroutine 替代命令模式

传统的命令模式通过封装请求和执行逻辑来解耦调用者和执行者。在 Go 中,可以直接通过 goroutine 和通道实现异步执行。

package mainimport "fmt"func main() {commandQueue := make(chan func(), 10)// 启动一个工作 goroutinego func() {for command := range commandQueue {command() // 执行命令}}()// 添加命令到队列commandQueue <- func() { fmt.Println("Command 1 executed") }commandQueue <- func() { fmt.Println("Command 2 executed") }close(commandQueue)
}

4. 总结

  • Go 的简单哲学:Go 通过接口、组合、goroutine 等特性,简化了很多传统设计模式的实现。
  • 减少复杂性:在 Go 中,我们可以直接使用函数、接口等轻量级工具来实现模式,而不是强依赖复杂的类和继承关系。
  • 场景驱动:Go 语言中的设计模式不是为了满足某种“模式理论”,而是为了适应具体的业务需求,强调代码的简洁和可读性。

Go 语言在使用设计模式时,更注重实践而不是形式。这种哲学让开发者能够专注于解决问题,而不是被复杂的模式实现所束缚。

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

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

相关文章

TCP 序列和确认号说明 | seq 和 ack 号计算方法

注&#xff1a;本文为 “TCP 序列” 相关文章合辑。 英文引文机翻未校。 TCP Sequence and Acknowledgement Numbers Explained TCP 序列和确认编号说明 TCP Sequence (seq) and Acknowledgement (ack) numbers help enable ordered reliable data transfer for TCP streams…

设计模式03:行为型设计模式之策略模式的使用情景及其基础Demo

1.策略模式 好处&#xff1a;动态切换算法或行为场景&#xff1a;实现同一功能用到不同的算法时和简单工厂对比&#xff1a;简单工厂是通过参数创建对象&#xff0c;调用同一个方法&#xff08;实现细节不同&#xff09;&#xff1b;策略模式是上下文切换对象&#xff0c;调用…

《数据思维》之数据可视化_读书笔记

文章目录 系列文章目录前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 数据之道&#xff0c;路漫漫其修远兮&#xff0c;吾将上下而求索。 一、数据可视化 最基础的数据可视化方法就是统计图。一个好的统计图应该满足四个标准&#xff1a;准确、有…

【AIGC】SYNCAMMASTER:多视角多像机的视频生成

标题&#xff1a;SYNCAMMASTER: SYNCHRONIZING MULTI-CAMERA VIDEO GENERATION FROM DIVERSE VIEWPOINTS 主页&#xff1a;https://jianhongbai.github.io/SynCamMaster/ 代码&#xff1a;https://github.com/KwaiVGI/SynCamMaster 文章目录 摘要一、引言二、使用步骤2.1 TextT…

持续集成 01|Gitee介绍、Pycharm使用Gitee

目录 一、理论 二、 git的简介与安装 三、Gitee 1、注册网易163邮箱 2、注册Gitee账号 3、git和gitee管理代码工作原理 三、PyCharm安装配置Gitee 四、Pycharm使用Gitee插件的五种场景 1、将 Gitee的新仓库 Checkout&#xff08;检出&#xff09;到 Pycharm中 2、推送…

maven 项目怎么指定打包后名字

在 Spring Boot 的 Maven 项目中&#xff0c;你可以通过配置 pom.xml 文件来指定打包后的文件名。具体步骤如下&#xff1a; 打开 pom.xml 文件&#xff1a;找到你的项目根目录下的 pom.xml 文件。 配置 finalName 属性&#xff1a;在 标签下&#xff0c;添加 属性来指定打包后…

Sprint Boot教程之五十八:动态启动/停止 Kafka 监听器

Spring Boot – 动态启动/停止 Kafka 监听器 当 Spring Boot 应用程序启动时&#xff0c;Kafka Listener 的默认行为是开始监听某个主题。但是&#xff0c;有些情况下我们不想在应用程序启动后立即启动它。 要动态启动或停止 Kafka Listener&#xff0c;我们需要三种主要方法…

前端常见的设计模式之【单例模式】

前端常见的设计模式&#xff1a; 单例模式观察者模式工厂模式适配器模式装饰器模式命令模式迭代器模式组合模式策略模式发布订阅模式 单例模式【创建型设计模式】&#xff1a; 单例模式是确保一个类只有一个实例&#xff0c;并提供一个全局访问点。这个模式非常适合那些需要…

C++|CRC校验总结

参考&#xff1a; Vector - CAPL - CRC算法介绍 开发工具 > CRC校验工具 文章目录 简介CRC-8CRC-16CRC-32 简介 循环冗余校验&#xff08;Cyclic Redundancy Check&#xff0c;简称CRC&#xff09;是一种数据校验算法&#xff0c;广泛用于检测数据传输或存储过程中的错误。…

# c语言:数组详解一

c语言&#xff1a;数组详解一 数组数组的概念引例&#xff1a;什么是数组数组的特征&#xff1a;下标&#xff08;索引&#xff09; 常用的数组按维度划分一维数组数组的定义&#xff1a;数组元素的访问数组的初始化**案例一、斐波拉契数列&#xff1a;****案例二、冒泡排序&am…

c#-Halcon入门教程——标定

Halcon代码 read_image (NinePointCalibration, D:/Desktop/halcon/ca74d-main/九点标定/NinePointCalibration.gif)rgb1_to_gray (NinePointCalibration, GrayImage)get_image_size (GrayImage, Width, Height) dev_display (GrayImage)* 获取当前显示的窗口句柄 dev_get_win…

语音识别的预训练模型

语音识别的预训练模型 语音识别模型 大致分为两类: 连接时序分类(Connectionist Temporal Classification, CTC):仅编码器(encoder-only)的模型,顶部带有线性分类(CTC)头序列到序列(Sequence-to-sequence, Seq2Seq):编码器-解码器(encoder-decoder)模型,编码器…

Kotlin 协程基础十 —— 协作、互斥锁与共享变量

Kotlin 协程基础系列&#xff1a; Kotlin 协程基础一 —— 总体知识概述 Kotlin 协程基础二 —— 结构化并发&#xff08;一&#xff09; Kotlin 协程基础三 —— 结构化并发&#xff08;二&#xff09; Kotlin 协程基础四 —— CoroutineScope 与 CoroutineContext Kotlin 协程…

4种革新性AI Agent工作流设计模式全解析

导读:AI Agent是指能够在特定环境中自主执行任务的人工智能系统,不仅接收任务,还自主制定和执行工作计划,并在过程中不断自我评估和调整,类似于人类在创造性任务中的思考和修正过程。AI Agent的四种关键设计模式是实现高效执行复杂任务的基础,共同构成了AI Agent的能力框…

Docker启动达梦 rman恢复

目录标题 1. 主库备份2. Docker启动备库3. 备库修改属组4. 开始恢复5. 连接数据库配置归档 & Open6. 检查数据 关于达梦数据库&#xff08;DMDBMS&#xff09;的主库备份、Docker启动备库、恢复备份以及配置归档和打开数据库的详细步骤。 1. 主库备份 # 使用达梦数据库备…

WPS excel使用宏编辑器合并 Sheet工作表

使用excel自带的工具合并Sheet表&#xff0c;我们会发现需要开通WPS会员才能使用合并功能&#xff1b; 那么WPS excel如何使用宏编辑器进行合并 Sheet表呢&#xff1f; 1、首先我们要看excel后缀是 .xlsx 还是 .xls &#xff1b;如果是.xlsx 那么 我们需要修改为 .xls 注…

MySQL之DDL语言

目录 一、数据库的基本操作 1、创建数据库 语法&#xff1a; 示例&#xff1a; 2、修改数据库 语法&#xff1a; 示例&#xff1a; 3、删除数据库 语法&#xff1a; 示例&#xff1a; 4、查询数据库 语法&#xff1a; 5、使用数据库 语法&#xff1a; 二、数据表…

js多种循环方法(通过循环进行判断的相关方法)

for&#xff1a;正常循环&#xff08;同步的循环&#xff09; break、continue终止循环 for (let index 0; index < array.length; index) {const element array[index];} forEach&#xff1a;正常循环&#xff08;异步的循环&#xff09; 通过try异常抛出终止循环&am…

RAG技术:是将知识库的文档和问题共同输入到LLM中

RAG技术 RAG技术是将知识库的文档和问题共同输入到LLM中 RAG技术是先从知识库中检索出与问题相关的文档片段,然后将这些检索到的文档片段与问题一起输入到LLM中进行回答。具体过程如下: 文本分块 由于LLM的上下文窗口有限,需要将长文本资料分割成较小的块,以便LLM能够有…

现代 CPU 的高性能架构与并发安全问题

现代 CPU 的设计&#xff08;如多级缓存、指令重排&#xff09;为了提升性能&#xff0c;引入了许多优化机制&#xff0c;但这些机制可能导致并发场景下的安全性问题。并发安全性主要体现在三个方面&#xff1a;原子性、有序性 和 可见性。这些问题在底层通过 CAS&#xff08;C…