写点什么

不会用“函数选项模式”的朋友看过来,这么写很优雅

作者:王中阳Go
  • 2023-02-28
    北京
  • 本文字数:3908 字

    阅读完需:约 13 分钟

不会用“函数选项模式”的朋友看过来,这么写很优雅

前言

通过这篇文章《为什么说Go的函数是”一等公民“》,我们了解到了什么是“一等公民”,以及都具备哪些特性,同时对函数的基本使用也更加深入。

本文重点介绍下 Go 设计模式之函数选项模式,它得益于 Go 的函数是“一等公民”,很好的一个应用场景,广泛被使用。

什么是函数选项模式

函数选项模式(Functional Options Pattern) ,也称为选项模式(Options Pattern),是一种创造性的设计模式,允许你使用接受零个或多个函数作为参数的可变构造函数来构建复杂结构。我们将这些函数称为选项,由此得名函数选项模式。

看概念有点太生硬难懂了,下面通过例子来讲解下怎么使用,由浅入深,通俗易懂。

怎么使用函数选项模式

一般水平

先来一个简单例子,这个Animal结构体,怎么构造出一个实例对象

type Animal struct {  Name   string  Age    int  Height int}
复制代码

通常的写法

func NewAnimal(name string, age int, height int) *Animal {  return &Animal{    Name:   name,    Age:    age,    Height: height,  }}
a1 := NewAnimal("小白兔", 5, 100)
复制代码

简单易懂,结构体有哪些属性字段,那么构造函数的参数,就相应做定义并传入

带来的问题

  1. 代码耦合度高:加属性字段,构造函数就得相应做修改,调用的地方全部都得改,势必会影响现有代码;

  2. 代码灵活度低:属性字段不能指定默认值,每次都得明确传入;

例如,现计划新加 3 个字段Weight体重CanRun是否会跑LegNum几条腿,同时要指定默认值CanRun=true、LegNum=4

新结构体定义

type Animal struct {  Name   string  Age    int  Height int  Weight int  CanRun bool  LegNum int}
复制代码

代码实现(函数加新参数定义,但默认值貌似实现不了,得调用构造函数时,明确传入):

func NewAnimal(name string, age int, height int, weight int, canRun bool, legNum int) *Animal {  return &Animal{    Name:   name,    Age:    age,    Height: height,    Weight: weight,    CanRun: canRun,    LegNum: legNum,  }}
a1 := NewAnimal("小白兔", 5, 100, 120, true, 4)
复制代码

后续逐步加新字段,这个构造函数就会被撑爆了,如果调用的地方越多,那么越伤筋动骨。

高阶水平

既然常规写法太 low,难以实现新需求,那么我们就来玩点高阶的,引出主题:函数选项模式

首先,需要先定义一个函数类型OptionFunc

type OptionFunc func(*Animal)
复制代码

然后,根据新结构体字段,定义With开头的函数,返回函数类型为OptionFunc的闭包函数,内部逻辑只需要实现更新对应字段值即可

func WithName(name string) OptionFunc {  return func(a *Animal) { a.Name = name }}
func WithAge(age int) OptionFunc {  return func(a *Animal) { a.Age = age }}
func WithHeight(height int) OptionFunc {  return func(a *Animal) { a.Height = height }}
func WithWeight(weight int) OptionFunc {  return func(a *Animal) { a.Weight = weight }}
func WithCanRun(canRun bool) OptionFunc {  return func(a *Animal) { a.CanRun = canRun }}
func WithLegNum(legNum int) OptionFunc {  return func(a *Animal) { a.LegNum = legNum }}
复制代码

再然后,优化构造函数的定义和实现(name 作为必传参数,其他可选,并且实现CanRunLegNum两个字段指定默认值)

func NewAnimal(name string, opts ...OptionFunc) *Animal {  a := &Animal{Name: name, CanRun: true, LegNum: 4}  for _, opt := range opts {    opt(a)  }  return a}
复制代码

最后,调用优化后的构造函数,快速实现实例的初始化。想要指定哪个字段值,那就调用相应的With开头的函数,完全做到可配置化、可插拔;不指定还支持了默认值

a2 := NewAnimal("大黄狗", WithAge(10), WithHeight(120))fmt.Println(a2)a3 := NewAnimal("大灰狼", WithHeight(200))fmt.Println(a3)
输出结果:&{大黄狗 10 120 0 true 4}&{大灰狼 0 200 0 true 4}
复制代码

带来的好处

  1. 高度的可配置化、可插拔,还支持默认值设定;

  2. 很容易维护和扩展;

  3. 容易上手,大幅降低新来的人试错成本;

开源项目中的实践案例

函数选项模式,不单单是我们业务代码中有使用,现在大量的标准库和第三库都在使用。

下面带着大家一块来看看,apollo 配置中心客户端第三库 shima-park/agollo[1],看看它是怎么玩的,怎么做配置初始化

核心代码

type Options struct {  AppID                      string               // appid  Cluster                    string               // 默认的集群名称,默认:default  DefaultNamespace           string               // Get时默认使用的命名空间,如果设置了该值,而不在PreloadNamespaces中,默认也会加入初始化逻辑中  PreloadNamespaces          []string             // 预加载命名空间,默认:为空  ApolloClient               ApolloClient         // apollo HTTP api实现  Logger                     Logger               // 日志实现类,可以设置自定义实现或者通过NewLogger()创建并设置有效的io.Writer,默认: ioutil.Discard  AutoFetchOnCacheMiss       bool                 // 自动获取非预设以外的Namespace的配置,默认:false  LongPollerInterval         time.Duration        // 轮训间隔时间,默认:1s  BackupFile                 string               // 备份文件存放地址,默认:.agollo  FailTolerantOnBackupExists bool                 // 服务器连接失败时允许读取备份,默认:false  Balancer                   Balancer             // ConfigServer负载均衡  EnableSLB                  bool                 // 启用ConfigServer负载均衡  RefreshIntervalInSecond    time.Duration        // ConfigServer刷新间隔  ClientOptions              []ApolloClientOption // 设置apollo HTTP api的配置项  EnableHeartBeat            bool                 // 是否允许兜底检查,默认:false  HeartBeatInterval          time.Duration        // 兜底检查间隔时间,默认:300s}
func newOptions(configServerURL, appID string, opts ...Option) (Options, error) {  var options = Options{    AppID:                      appID,    Cluster:                    defaultCluster,    ApolloClient:               NewApolloClient(),    Logger:                     NewLogger(),    AutoFetchOnCacheMiss:       defaultAutoFetchOnCacheMiss,    LongPollerInterval:         defaultLongPollInterval,    BackupFile:                 defaultBackupFile,    FailTolerantOnBackupExists: defaultFailTolerantOnBackupExists,    EnableSLB:                  defaultEnableSLB,    EnableHeartBeat:            defaultEnableHeartBeat,    HeartBeatInterval:          defaultHeartBeatInterval,  }  for _, opt := range opts {    opt(&options)  }
  //...省略
  return options, nil}
type Option func(*Options)
//一系列函数作为选项func PreloadNamespaces(namespaces ...string) Option {  return func(o *Options) {    o.PreloadNamespaces = append(o.PreloadNamespaces, namespaces...)  }}func AutoFetchOnCacheMiss() Option {  return func(o *Options) {    o.AutoFetchOnCacheMiss = true  }}//...
复制代码

玩法

  1. 使用 Options 结构体,定义出 apollo 需要使用到的所有配置字段;

  2. 定义一系列函数作为选项,对配置字段做初始化设置(例如,设置容灾文件路径、预加载的 namespace、轮训间隔时间等等);

  3. 构造函数里初始化一个 Options 的实例对象,并且根据传入的函数选项,进行配置字段的更新,最终返回这个实例对象;

  4. 获取到实例对象,调用相应的方法做相应的操作。

总结

由浅入深的讲解了下实例对象初始化一般写法和高阶写法。用好这个高阶写法(函数选项模式),让代码更优雅。


还不会使用的 Gopher,赶紧学起来,用起来。

相关资料

[1]

http://github.com/shima-park/agollo: https://link.juejin.cn?target=http%3A%2F%2Fgithub.com%2Fshima-park%2Fagollo


一起学习

我的文章首发在我的公众号: 程序员升职加薪之旅,欢迎大家关注,第一时间阅读我的文章。


也欢迎大家关注我,点赞、留言、转发。你的支持,是我更文的最大动力!

发布于: 2023-02-28阅读数: 42
用户头像

王中阳Go

关注

靠敲代码在北京买房的程序员 2022-10-09 加入

【微信】wangzhongyang1993【公众号】程序员升职加薪之旅【成就】InfoQ专家博主👍掘金签约作者👍B站&掘金&CSDN&思否等全平台账号:王中阳Go

评论 (4 条评论)

发布
用户头像
可以
4 小时前 · 广东
回复
1
4 小时前 · 广东
回复
用户头像
好文,推荐了
2023-02-28 19:16 · 江苏
回复
用户头像
函数选项模式(Functional Options Pattern) ,也称为选项模式(Options Pattern),是一种创造性的设计模式,允许你使用接受零个或多个函数作为参数的可变构造函数来构建复杂结构。我们将这些函数称为选项,由此得名函数选项模式。
2023-02-28 11:07 · 北京
回复
没有更多了
不会用“函数选项模式”的朋友看过来,这么写很优雅_Go_王中阳Go_InfoQ写作社区