一文带你搞懂Go语言函数选项模式,Go函数一等公民。

news/2024/4/27 6:01:19/文章来源:https://blog.csdn.net/w425772719/article/details/129257018

前言

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

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

什么是函数选项模式

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

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

怎么使用函数选项模式

一般水平

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

type Animal struct {Name   stringAge    intHeight 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   stringAge    intHeight intWeight intCanRun boolLegNum 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,看看它是怎么玩的,怎么做配置初始化

核心代码

type Options struct {AppID                      string               // appidCluster                    string               // 默认的集群名称,默认:defaultDefaultNamespace           string               // Get时默认使用的命名空间,如果设置了该值,而不在PreloadNamespaces中,默认也会加入初始化逻辑中PreloadNamespaces          []string             // 预加载命名空间,默认:为空ApolloClient               ApolloClient         // apollo HTTP api实现Logger                     Logger               // 日志实现类,可以设置自定义实现或者通过NewLogger()创建并设置有效的io.Writer,默认: ioutil.DiscardAutoFetchOnCacheMiss       bool                 // 自动获取非预设以外的Namespace的配置,默认:falseLongPollerInterval         time.Duration        // 轮训间隔时间,默认:1sBackupFile                 string               // 备份文件存放地址,默认:.agolloFailTolerantOnBackupExists bool                 // 服务器连接失败时允许读取备份,默认:falseBalancer                   Balancer             // ConfigServer负载均衡EnableSLB                  bool                 // 启用ConfigServer负载均衡RefreshIntervalInSecond    time.Duration        // ConfigServer刷新间隔ClientOptions              []ApolloClientOption // 设置apollo HTTP api的配置项EnableHeartBeat            bool                 // 是否允许兜底检查,默认:falseHeartBeatInterval          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,赶紧学起来,用起来。

文章首发

我的文章会首发在我的公众号:程序员升职加薪之旅,欢迎大家关注,第一时间收到最新内容。

一起学习

我的所有文章都会首发在我的 学习小圈子 ,欢迎加入我们,一起学习进步,一起升职加薪。

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

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

相关文章

《C++ Primer Plus》(第6版)第5章编程练习

《C Primer Plus》(第6版)第5章编程练习《C Primer Plus》(第6版)第5章编程练习1. 计算闭区间内的整数和2. 重新编写程序清单5.43. 累加4. 投资价值5. 销售情况6. 销售情况27. 汽车8. 销售情况29. 销售情况210. 销售情况2《C Prim…

【技术美术图形部分】简述主流及新的抗锯齿技术

电脑的世界里没有曲线,都是三角面组成一个个模型的,因此一定会出现走样(锯齿)的情况,只是严重与否的问题,而AA也是实时渲染最难解决的问题之一。 Sampling&Artifacts Lecture 06 Rasterization 2 (An…

MAML算法详解(元学习)

文章目录回顾元学习MAML算法MAML和预训练模型的区别数学推导MAML实施细节总结回顾元学习 元学习的基本知识参考这篇博客元学习和机器学习的对比 MAML算法 学习初始化参数,所有任务的初始化的参数都是一样的 MAML和预训练模型的区别 MAML使用的是ϕ\phiϕ…

阶段十:总结专题(第六章:缓存篇)

阶段十:总结专题(第六章:缓存篇)Day-第六章:缓存篇1. Redis 数据类型**String****List****Hash****Sorted Set**2. keys 命令问题3. 过期 key 的删除策略4. Redis 持久化**AOF 持久化****AOF 重写****RDB 持久化****混…

Python 中 openpyxl 模块封装,读写 Excel 文件中自动化测试用例数据

只有测试数据和错误提示信息不同,其他代码都是一样的,不这样不易修改数据和维护,会有两点痛点 1.代码冗余极其严重, 程序可读性不佳 2.程序拓展性很差 往往我们在自动化测试汇总,会将数据放在 Excel 文件、CSV文件、数据库 Py…

Python-scatter散点图及颜色大全

# -*- coding: utf-8 -*- import numpy as np import matplotlib.pyplot as pltplt.rcParams[font.sans-serif][SimHei] plt.rcParams[axes.unicode_minus] False #matplotlib画图中中文显示会有问题,需要这两行设置默认字体plt.xlabel(X) plt.ylabel(Y) plt.xlim…

【IP技术】ipv4和ipv6是什么?

IPv4和IPv6是两种互联网协议,用于在互联网上标识和寻址设备。IPv4(Internet Protocol version 4)是互联网协议的第四个版本,是当前广泛使用的互联网协议。IPv4地址由32位二进制数构成,通常表示为4个十进制数&#xff0…

大数据技术之Hive(四)分区表和分桶表、文件格式和压缩

一、分区表和分桶表1.1 分区表partitionhive中的分区就是把一张大表的数据按照业务需要分散的存储到多个目录,每个目录就称为该表的一个分区。在查询时通过where子句中的表达式选择式选择查询所需要的分区,这样的查询效率辉提高很多。1.1.1 分区表基本语…

2023年蜂巢科技最新面试题

2023年蜂巢科技最新面试题 bio与nio的区别 bio同步阻塞io:在此种⽅式下,⽤户进程在发起⼀个IO操作以后,必须等待IO操作的完成,只有当真正完成了IO操作以后,⽤户进程才能运⾏。JAVA传统的IO模型属于此种⽅式&#xff0…

flink常用算子介绍

flink任务中【Transformation 数据转换】是对数据进行操作,有 Map、FlatMap、Filter、KeyBy 、Reduce 、Fold 、Aggregations、Window 、WindowAll 、Union 、Window join 、Split 、Select 、Project 等,通过对数据的操作,转换成想要的数据&…

HttpRunnerManager部署

基于HttpRunner的接口自动化测试平台: HttpRunner, djcelery and Django_. HttpRunner手册: http://cn.httprunner.org/git地址:httprunner/HttpRunnerManager: 基于 HttpRunner 的 Web 测试平台,已停止维护。 (github.com)部署机器:linux部署…

vue3+rust个人博客建站日记4-Vditor搞定MarkDown

即然是个人博客,那么绝对不能丢给自己一个大大的输入框敷衍了事。如果真是这样,现在就可以宣布项目到此结束了。如今没人享受用输入框写博客。作为一个有追求的程序员,作品就要紧跟潮流。 后来,Markdown 的崛起逐步改变了大家的排…

华为OD机试题,用 Java 解【火星文计算 2】问题

最近更新的博客 华为OD机试题,用 Java 解【停车场车辆统计】问题华为OD机试题,用 Java 解【字符串变换最小字符串】问题华为OD机试题,用 Java 解【计算最大乘积】问题华为OD机试题,用 Java 解【DNA 序列】问题华为OD机试 - 组成最大数(Java) | 机试题算法思路 【2023】使…

跑步戴哪种蓝牙耳机比较好,五款适合跑步的蓝牙耳机推荐

跑步当中佩戴的蓝牙耳机,佩戴舒适度以及牢固度是我们首要关注的,耳机的材质还有耳机的防水能力,都是十分需要注意的方面。那具体在挑选运动耳机当中需要如何选择呢?下面收集的一些在运动当中比较好用的运动蓝牙耳机分享给大家。1、…

华为OD机试题,用 Java 解【机器人走迷宫】问题

最近更新的博客 华为OD机试题,用 Java 解【停车场车辆统计】问题华为OD机试题,用 Java 解【字符串变换最小字符串】问题华为OD机试题,用 Java 解【计算最大乘积】问题华为OD机试题,用 Java 解【DNA 序列】问题华为OD机试 - 组成最大数(Java) | 机试题算法思路 【2023】使…

力扣每日一题(2023年2月)

2023年2月期每日一题第一天 (2325. 解密消息)第十六天(2341. 数组能形成多少数对)第十七天 (1139. 最大的以 1 为边界的正方形)第十八天 (1237. 找出给定方程的正整数解)第十九天 &a…

【再临数据结构】Day1. 稀疏数组

前言 这不单单是稀疏数组的开始,也是我重学数据结构的开始。因此,在开始说稀疏数组的具体内容之前,我想先说一下作为一个有着十余年“学龄”的学生,所一直沿用的一个学习方法:3W法。我认为,只有掌握了正确的…

Apache Hive入门

文章目录一、Apache Hive概述1.1、什么是Hive1.2、使用Hive原因1.3、Hive和Hadoop关系二、Hive功能思想2.1、映射信息记录2.2、SQL语法解析、编译三、Hive架构、组件3.1、Hive架构图3.2Hive组件四、Hive常用操作4.1、数据类型4.1.1、基本数据类型4.1.2、集合数据类型4.2、数据库…

本地新项目上传到git的详细步骤

前提:你本地的项目目录里要记得添加.gitignore忽略文件,免得把一些无用的文件提交,内容如下,可直接粘贴: # Created by .ignore support plugin (hsz.mobi) ### Java template # Compiled class file *.class# Log fi…

2023-02-28 mmap的原理及使用-思考

摘要: 最近在使用mmap解决数据库内存占用损耗过高导致OOM的问题, 不得不说在有些场景下mmap是非常有用. 本文主要涉及一些对mmap的思考. mmap本身的思考: mmap和文件系统的交互规则是什么mmap中给进程虚拟内存映射的文件上的部分,是什么? 为什么是页缓存? 有没有文件缓存?…