迹忆客 专注技术分享

当前位置:主页 > 学无止境 > 编程语言 > Go >

Golang 函数选项模式

作者:迹忆客 最近更新:2023/01/16 浏览次数:

Go (Golang) 函数选项模式是一种方式,一种在 Go 中构建结构的模式,它通过设计一组非常具有表现力和灵活性的 API 来帮助配置和初始化你的结构。 让我们看一下代码片段,看看我们可以使用哪些选项,以及功能选项模式如何以及何时对我们有用。


示例:在 Go 中构建服务器包

在此示例中,我们查看 Go 中的服务器包,但它可以是第三方客户端使用的任何东西,如自定义 SDK 或记录器库。

package server

type Server {
  host string
  port int
}

func New(host string, port int) *Server {
  return &Server{host, port}
}

func (s *Server) Start() error {
  // todo
}

这是客户端如何导入和使用我们的服务器包

package main

import (
  "log"
  
  "github.com/acme/pkg/server"
)

func main() {
  svr := server.New("localhost", 1234)
  if err := svr.Start(); err != nil {
    log.Fatal(err)
  }
}

现在,鉴于这种情况,我们如何扩展服务器的配置选项? 有几个选项

  • 为每个不同的配置选项声明一个新的构造函数
  • 定义一个新的 Config 结构来保存配置信息
  • 使用功能选项模式

让我们一一探讨这 3 个示例,并分析每个示例的优缺点。


选项 1:为每个配置选项声明一个新的构造函数

如果你知道配置选项不会很幸运地发生变化并且配置选项很少,那么这可能是一个很好的方法。 因此,只需为每个不同的配置选项创建新方法就很容易了。

package server

type Server {
  host string
  port int
  timeout time.Duration
  maxConn int
}

func New(host string, port int) *Server {
  return &Server{host, port, time.Minute, 100}
}

func NewWithTimeout(host string, port int, timeout time.Duration) *Server {
  return &Server{host, port, timeout}
}

func NewWithTimeoutAndMaxConn(host string, port int, timeout time.Duration, maxConn int) *Server {
  return &Server{host, port, timeout, maxConn}
}

func (s *Server) Start() error {
  // todo
}

以及下面的相关客户端实现

package main

import (
  "log"
  
  "github.com/acme/pkg/server"
)

func main() {
  svr := server.NewWithTimeoutAndMaxConn("localhost", 1234, 30*time.Second, 10)
  if err := svr.Start(); err != nil {
    log.Fatal(err)
  }
}

当配置选项的数量增加或经常更改时,这种方法不是很灵活。 我们还需要为每个新的配置选项或配置选项集创建新的构造函数。


选项 2:使用自定义 Config 结构

这是最常见的方法,当有很多选项需要配置时可以很好地工作。 我们可以创建一个名为“Config”的新导出类型,其中包含服务器的所有配置选项。 这可以在不破坏服务器构造函数 API 的情况下轻松扩展。 添加新选项或删除旧选项时,我们不必更改其定义

package server

type Server {
  cfg Config
}

type Config struct {
  host string
  port int
  timeout time.Duration
  maxConn int
}

func New(cfg Config) *Server {
  return &Server{cfg}
}

func (s *Server) Start() error {
  // todo
}

以及下面使用新的 Config 结构的相关客户端实现

package main

import (
  "log"
  
  "github.com/acme/pkg/server"
)

func main() {
  svr := server.New(server.Config{"localhost", 1234, 30*time.Second, 10})
  if err := svr.Start(); err != nil {
    log.Fatal(err)
  }
}

这种方法非常灵活,允许我们为我们的服务器(或 SDK 客户端或您正在构建的任何东西)定义固定类型(server.Config)和一组稳定的 API 来配置我们的服务器,如 server.New(cfg server.Config)。 唯一的问题是,当添加新选项或删除旧选项时,我们仍然需要对 Config 结构的结构进行重大更改。 但这仍然是迄今为止最好的和更可用的选择。


选项 3:功能选项模式

这个选项配置问题的一个更好的替代方案是功能选项设计模式。 我们可能以前在 Go 项目中看到或听说过功能选项模式,但在这个例子中,我们将详细分解它的结构和特征。

package server

type Server {
  host string
  port int
  timeout time.Duration
  maxConn int
}

func New(options ...func(*Server)) *Server {
  svr := &Server{}
  for _, o := range options {
    o(svr)
  }
  return svr
}

func (s *Server) Start() error {
  // todo
}

func WithHost(host string) func(*Server) {
  return func(s *Server) {
    s.host = host
  }
}

func WithPort(port int) func(*Server) {
  return func(s *Server) {
    s.port = port
  }
}

func WithTimeout(timeout time.Duration) func(*Server) {
  return func(s *Server) {
    s.timeout = timeout
  }
}

func WithMaxConn(maxConn int) func(*Server) {
  return func(s *Server) {
    s.maxConn = maxConn
  }
}

以及下面使用新功能选项模式的相关客户端实现

package main

import (
  "log"
  
  "github.com/acme/pkg/server"
)

func main() {
  svr := server.New(
    server.WithHost("localhost"),
    server.WithPort(8080),
    server.WithTimeout(time.Minute),
    server.WithMaxConn(120),
  )
  if err := svr.Start(); err != nil {
    log.Fatal(err)
  }
}

功能选项模式允许我们为我们的服务器的每个和任何可能的配置定义一个固定的类型签名,使用 func(*Server) 类型签名我们可以创建任何选项传递给服务器。 默认情况下,我们的选项也是可选的,因此可以轻松交换任何选项而不会出现任何重大问题。 鉴于类型定义的表现力设计和自动记录性质,这种方法也很好,每个方法都为我们的服务器定义选项和选项类型。

转载请发邮件至 1244347461@qq.com 进行申请,经作者同意之后,转载请以链接形式注明出处

本文地址:

相关文章

Golang 中的零值 Nil

发布时间:2023/04/27 浏览次数:166 分类:Go

本篇文章介绍 nil 在 Golang 中的含义,nil 是 Go 编程语言中的零值,是众所周知且重要的预定义标识符。

Golang 中的 Lambda 表达式

发布时间:2023/04/27 浏览次数:93 分类:Go

本篇文章介绍如何在 Golang 中创建 lambda 表达式。Lambda 表达式似乎不存在于 Golang 中。 函数文字、lambda 函数或闭包是匿名函数的另一个名称。

在 Go 中使用断言

发布时间:2023/04/27 浏览次数:181 分类:Go

本篇文章介绍了 assert 在 GoLang 中的使用。在 Go 语言中使用断言:GoLang 不提供对断言的任何内置支持,但我们可以使用来自 Testify API 的广泛使用的第三方包断言。

Go 中的随机数生成

发布时间:2023/04/27 浏览次数:114 分类:Go

本篇文章介绍如何在 Go 语言中使用随机数生成功能。Go 中的随机数生成 Go 语言为随机数生成功能提供内置支持。 内置包 math 有方法 rand(),用于随机数生成。

GoLang 电子邮件验证器

发布时间:2023/04/27 浏览次数:195 分类:Go

本篇文章介绍如何在 Go 语言中验证电子邮件。电子邮件需要特定格式; 否则,它们将无法工作。

卸载 GoLang

发布时间:2023/04/27 浏览次数:183 分类:Go

本篇文章介绍如何在不同的操作系统上卸载 GoLang。卸载 Go 语言 只需几个简单的步骤即可卸载 GoLang。 这些步骤对于不同的操作系统是不同的。

在 Go 中使用 Electron API 创建 GUI

发布时间:2023/04/27 浏览次数:124 分类:Go

本篇文章介绍如何在 Go 语言中使用 Electron API 创建 GUI。Electron API 或 Astilectron 用于为 GoLang 创建 GUI。

在 GoLang 中安装包

发布时间:2023/04/27 浏览次数:122 分类:Go

使用 Go 语言的 get 命令安装所需的包非常容易。 Go 语言提供了多种命令来执行某些任务,get 就是其中之一。

扫一扫阅读全部技术教程

社交账号
  • https://www.github.com/onmpw
  • qq:1244347461

最新推荐

教程更新

热门标签

扫码一下
查看教程更方便