这些Coding套路你不会还不知道吧?

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 这些Coding套路你不会还不知道吧?

对于一名程序员来说,编码进阶是成为优秀工程师非常重要的一步,它可以让我们更加熟练地掌握编程,深入理解数据结构和算法,从而更好地完成复杂的任务,提高工作效率。而我认为熟练使用设计模式就是编码进阶的最好方式之一,下面就用一篇文章来分享下Go开发中常用的设计模式。

  • 1️⃣全局单一实例:单例模式
  • 🚗获取对象超简单:工厂模式
  • 📋重复代码太多?试试:模板方法模式
  • 👻接口对应功能不知道怎么维护?这里有:策略模式
  • 🎈独特好玩的Functional Options模式

What?Why?

从两个问题开始今天的分享:

设计模式是什么?

简单来讲,就两个字:套路,编码的套路,在我们写代码的时候这个套路可以不用,但是不能不知道。

为什么要使用设计模式?

设计模式是一种被反复使用的,针对软件设计中常见问题的可复用解决方案。它们提供了一种经过验证的方式来解决复杂的设计问题,并帮助开发人员编写更加清晰、可维护和可扩展的代码

使用设计模式的好处包括:

  1. 提高代码的可读性和可维护性。通过使用设计模式,开发人员可以将复杂的问题分解成更小、更易于管理的部分,并且可以将这些部分组织成一致的、易于理解的代码结构。
  2. 提高代码的可重用性。设计模式提供了一种标准化的解决方案,可以让相同的功能代码在多个地方重复使用,从而避免了重复编写相同的代码,减少了开发时间和成本。
  3. 提高代码的灵活性和扩展性。设计模式允许开发人员在不改变现有代码的情况下,轻松地添加新的功能或修改现有的功能。

常见的设计模式实践

1 全局单一实例:单例模式

首先,单例模式一般用于以下业务场景:

(1)整个程序的运行中只允许有一个类的实例。

(2)创建对象时耗时过多或者耗资源过多,但又经常用到的。

(3)需要全局访问并保证线程安全的对象。

(4)需要保持状态的对象。

具体场景:

数据库连接对象。我们在具体的项目中往往每种数据库驱动只会使用它的一个实例,因此在这里我们就能够使用单例模式。

代码:

import "sync"
type MysqlConn struct {
  Addr string
}
var (
  mysqlConn *MysqlConn
  once = sync.Once{}
)
func GetMySQLConn() *MysqlConn {
  once.Do(func() {
    mysqlConn = &MysqlConn{Addr: "127.0.0.1"}
  })
  return mysqlConn
}
2 获取对象超简单:工厂模式

工厂模式通常应用于以下场景:

(1)当一个系统需要灵活地配置一组对象,并且需要动态地选择其中的一个时。

(2)当一个类需要频繁地创建和销毁时,可以使用工厂模式来提高性能。

具体场景:

简单来讲就是想要获取一个实例,但是这个实例的属性可能会随着参数的变化而具有不同的形态,还可以用数据库连接来举例,我们有多个服务器可以进行连接,但是每次只能连接一个,而且连接的时候可以填写自己的认证账密。

代码:

type DB struct {
   Addr, UserName, Passwd string
}
func NewMysqlConn(addr, userName, passwd string) *DB {
   return &DB{
      Addr:     addr,
      UserName: userName,
      Passwd:   passwd,
   }
}
3 重复代码太多?试试:模板方法模式

模板方法模式通常应用于以下场景:

(1)当一个算法的步骤中有一部分是不变的,而另一部分是需要根据不同的条件进行变化时,可以使用模板方法模式来实现。

(2)当一个类的子类需要实现一个接口的不同版本时,可以使用模板方法模式来实现。

具体场景:

做水果酸奶,水果或其他食材是统一的抽象,而火龙果、芒果、饼干都是具体,因为不管用什么水果或食材来做,做法都是如出一辙的,所以做法也是一个抽象,这就好比一个模板,对,就是模版方法模式。

代码:

import (
   "fmt"
   "testing"
)
type MakeYogurtTemplate interface {
  CreateYogurt() //准备好酸奶
  CutFruit()     //切水果
  Merge()        //放在一起搅拌
  Optimize()     //调制味道
  Eating()       //开吃
  Do()
}
type DragonFruit struct {
}
func (d *DragonFruit) CreateYogurt() {
   fmt.Println("用鲜奶和乳酸菌发酵好酸奶")
}
func (d *DragonFruit) CutFruit() {
   fmt.Println("把火龙果切成块块")
}
func (d *DragonFruit) Merge() {
   fmt.Println("放在一起进行搅拌")
}
func (d *DragonFruit) Optimize() {
   fmt.Println("调制出自己喜欢的味道")
}
func (d *DragonFruit) Eating() {
   fmt.Println("开吃")
}
func (d *DragonFruit) Do() {
   d.CreateYogurt()
   d.CutFruit()
   d.Merge()
   d.Optimize()
   d.Eating()
}
func TestDragonFruit(t *testing.T) {
   d := &DragonFruit{}
   d.Do()
}
type Mango struct {
}
func (m *Mango) CreateYogurt() {
   fmt.Println("用鲜奶和乳酸菌发酵好酸奶")
}
func (m *Mango) CutFruit() {
   fmt.Println("把芒果切成块块")
}
func (m *Mango) Merge() {
   fmt.Println("放在一起进行搅拌")
}
func (m *Mango) Optimize() {
   fmt.Println("调制出自己喜欢的味道")
}
func (m *Mango) Eating() {
   fmt.Println("开吃")
}
func (m *Mango) Do() {
   m.CreateYogurt()
   m.CutFruit()
   m.Merge()
   m.Optimize()
   m.Eating()
}
func TestMango(t *testing.T) {
   m := &Mango{}
   m.Do()
}
4 接口对应功能不知道怎么维护?这里有:策略模式

策略模式通常在一个系统中需要多个算法,并且这些算法需要在不同的时间或条件下使用不同的算法时,可以使用策略模式来实现。

具体场景:

对于新手程序员同学经常会有这样的选择,到底是学Go还是学Java,这显然是两种不同的策略选择,但是不管是学习哪一个的基本流程都是差不多的,比如都是先准备学习资料,然后在学习和实践,因此可以基于此使用策略模式来形容。

代码:

策略抽象

type Learn interface {
   PrepareData()
   DoLearn()
   Play()
}
func LearnLang(l Learn) {
   l.PrepareData()
   l.DoLearn()
   l.Play()
}

策略1

import "fmt"
type LikeGo struct {
}
func (g *LikeGo) PrepareData() {
   fmt.Println("准备Go资料")
}
func (g *LikeGo) DoLearn() {
   fmt.Println("学习Go")
}
func (g *LikeGo) Play() {
   fmt.Println("玩转Go语言")
}

策略2

import "fmt"
type LikeJava struct {
}
func (g *LikeJava) PrepareData() {
   fmt.Println("准备Java资料")
}
func (g *LikeJava) DoLearn() {
   fmt.Println("学习Java")
}
func (g *LikeJava) Play() {
   fmt.Println("玩Java")
}

执行策略

import "testing"
func TestLearn(t *testing.T) {
   likeGo := &LikeGo{}
   LearnLang(likeGo)
}
5 独特好玩的Functional Options模式

Functional Options模式通常在当一个对象或函数具有多个参数时,这些参数可能会有不同的默认值或取值范围。通过将它们作为函数的选项传递,可以更灵活地控制函数的行为

具体场景:

gRPC服务进行实例化的时候有些参数可以不同填,即选填,之后在源码内部会使用默认的值,于是我们就可以使用以下的方式进行处理。

代码:

import "time"
type RpcServer struct {
   Name    string
   MaxConn int
   Address []string
   TimeOut time.Duration
}
type RpcServerOption func(server *RpcServer)
func WithName(name string) RpcServerOption {
   return func(server *RpcServer) {
      server.Name = name
   }
}
func WithMaxConn(max int) RpcServerOption {
   return func(server *RpcServer) {
      server.MaxConn = max
   }
}
func WithAddress(addr []string) RpcServerOption {
   return func(server *RpcServer) {
      server.Address = addr
   }
}
func WithTimeOut(timeout time.Duration) RpcServerOption {
   return func(server *RpcServer) {
      server.TimeOut = timeout
   }
}
func NewRpcServer(opts ...RpcServerOption) *RpcServer {
   server := &RpcServer{}
   for _, opt := range opts {
      opt(server)
   }
   return server
}

实例化:

import (
   "fmt"
   "testing"
   "time"
)
func TestCreateRpcServerByOptions(t *testing.T) {
   rpcServer := NewRpcServer(
      WithAddress([]string{"127.0.0.1"}),
      WithName("rpcServer"),
      WithMaxConn(1),
      WithTimeOut(time.Second),
   )
   fmt.Println(*rpcServer)
}

小总结

本文主要介绍了Go开发中常用的设计模式,包括全局单一实例:单例模式、工厂模式、模板方法模式、策略模式和Functional Options模式。这些设计模式可以帮助我们更好地组织代码,提高代码的可读性和可重用性。

总之,掌握这些设计模式对于提高Go程序员的编码能力非常有帮助,可以让我们在编写代码时更加得心应手,同时也能提高代码的质量和可维护性。

相关实践学习
每个IT人都想学的“Web应用上云经典架构”实战
本实验从Web应用上云这个最基本的、最普遍的需求出发,帮助IT从业者们通过“阿里云Web应用上云解决方案”,了解一个企业级Web应用上云的常见架构,了解如何构建一个高可用、可扩展的企业级应用架构。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。   相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情: https://www.aliyun.com/product/rds/mysql 
相关文章
|
5月前
|
Docker 容器
熟悉Docker容器管理命令:start、stop与restart详细使用指南
掌握这些Docker容器管理命令对于维护应用程序的正常运行至关重要。在实际操作中,应注意容器配置、关联资源以及日志等信息,确保各项操作都能够顺畅并且安全地执行。
640 0
|
机器学习/深度学习 人工智能 数据挖掘
【AI 生成式】半监督学习和自监督学习的概念
【5月更文挑战第4天】【AI 生成式】半监督学习和自监督学习的概念
|
安全 网络安全 网络架构
什么是端口转发?什么是端口映射?如何设置端口映射
端口映射与端口转发是网络配置中两个常被混淆的概念。端口映射是指将外部网络请求通过路由器转发至内部网络特定主机的过程,增强了内网安全性。而端口转发则是指路由器依据端口将外部请求定向至具体设备,实现内外网通信。两者虽相似,但应用场景和原理有所不同。通过工具如花生壳,可轻松设置端口映射,实现外网访问内网服务。
2540 1
|
消息中间件 JSON Java
Springboot支付宝沙箱支付---完整详细步骤
Springboot支付宝沙箱支付---完整详细步骤
2760 1
|
缓存
【POI】导出xls文件报错:The maximum number of cell styles was exceeded. You can define up to 4000 styles in a .xls workbook
使用POI导出xls文件,由于数据过多,导致导出xls报错如下: The maximum number of cell styles was exceeded. You can define up to 4000 styles in a .xls workbook   原因: 代码中创建 HSSFCellStyle cellStyle = hssfWorkbook.createCellStyle(); 次数过多,导致报错。
7013 0
|
存储 安全 文件存储
开发者如何使用文件存储NAS
【10月更文挑战第7天】开发者如何使用文件存储NAS
1062 0
|
开发工具 Android开发
Android平台GB28181设备接入端预置位查询(PresetQuery)探讨和技术实现
之前blog介绍了GB28181云台控制(PTZCmd)相关,本文主要是介绍下GB28181预置位查询。
378 0
|
机器学习/深度学习 数据采集 分布式计算
机器学习流水线的六个步骤
【5月更文挑战第30天】机器学习应用通过构建流水线实现,简化大数据需求和学习任务的处理,使用户能专注核心任务而非基础设施。
|
缓存 NoSQL 关系型数据库
MySQL缓存策略(一致性问题、数据同步以及缓存故障)
MySQL缓存策略(一致性问题、数据同步以及缓存故障)
382 1
解决:org.springframework.web.method.annotation.MethodArgumentTypeMismatchExceptio
解决:org.springframework.web.method.annotation.MethodArgumentTypeMismatchExceptio
640 0