有时候一个函数会有不少参数,为了方便函数的使用,咱们会给但愿给一些参数设定默认值,调用时只须要传与默认值不一样的参数便可,相似于 python 里面的默认参数和字典参数,虽然 golang 里面既没有默认参数也没有字典参数,可是咱们有选项模式python
在这以前,首先须要介绍一下可变长参数列表,顾名思义,就是参数的个数不固定,能够是一个也能够是多个,最典型的用法就是标准库里面的 fmt.Printf
,语法比较简单,以下面例子实现任意多个参数的加法git
func add(nums ...int) int { sum := 0 for _, num := range nums { sum += num } return sum } So(add(1, 2), ShouldEqual, 3) So(add(1, 2, 3), ShouldEqual, 6)
在类型前面加 ...
来表示这个类型的变长参数列表,使用上把参数当成 slice
来用便可github
假设咱们要实现这样一个函数,这个函数接受5个参数,三个 string
(其中第一个参数是必填参数),两个 int
,这里功能只是简单输出这个参数,因而咱们能够简单用以下代码实现golang
func MyFunc1(requiredStr string, str1 string, str2 string, int1 int, int2 int) { fmt.Println(requiredStr, str1, str2, int1, int2) } // 调用方法 MyFunc1("requiredStr", "defaultStr1", "defaultStr2", 1, 2)
这种实现比较简单,可是同时传入参数较多,对调用方来讲,使用的成本就会比较高,并且每一个参数的具体含义这里并不清晰,很容易出错函数
那选项模式怎么实现这个需求呢?先来看下最终的效果ui
MyFunc2("requiredStr") MyFunc2("requiredStr", WithOptionStr1("mystr1")) MyFunc2("requiredStr", WithOptionStr2AndInt2("mystr2", 22), WithOptionInt1(11))
如上面代码所示,你能够根据本身的需求选择你须要传入的参数,大大简化了函数调用的复杂度,而且每一个参数都有了清晰明确的含义设计
那怎么实现上面的功能呢code
首先定义可选项和默认值,这里有4个可选项,第一个参数为必填项rpc
type MyFuncOptions struct { optionStr1 string optionStr2 string optionInt1 int optionInt2 int } var defaultMyFuncOptions = MyFuncOptions{ optionStr1: "defaultStr1", optionStr2: "defaultStr2", optionInt1: 1, optionInt2: 2, }
这些 With 方法看起来有些古怪,接受一个选项参数,返回一个选项方法,而选项方法以选项做为参数负责修改选项的值,若是没看明白不要紧,能够先看函数功能如何实现get
type MyFuncOption func(options *MyFuncOptions) func WithOptionStr1(str1 string) MyFuncOption { return func(options *MyFuncOptions) { options.optionStr1 = str1 } } func WithOptionInt1(int1 int) MyFuncOption { return func(options *MyFuncOptions) { options.optionInt1 = int1 } } func WithOptionStr2AndInt2(str2 string, int2 int) MyFuncOption { return func(options *MyFuncOptions) { options.optionStr2 = str2 options.optionInt2 = int2 } }
这里咱们让 optionStr2 和 optionInt2 合并一块儿设置,实际应用场景中能够用这种方式将相关的参数放到一块儿设置
func MyFunc2(requiredStr string, opts ...MyFuncOption) { options := defaultMyFuncOptions for _, o := range opts { o(&options) } fmt.Println(requiredStr, options.optionStr1, options.optionStr2, options.optionInt1, options.optionInt2) }
使用 With 方法返回的选项方法做为参数列表,用这些方法去设置选项
从这里能够看到,为了实现选项的功能,咱们增长了不少的代码,实现成本相对仍是较高的,因此实践中须要根据本身的业务场景去权衡是否须要使用。我的总结知足下面条件能够考虑使用选项模式
在 golang 的不少开源项目里面也用到了选项模式,好比 grpc 中的 rpc 方法就是采用选项模式设计的,除了必填的 rpc 参数外,还能够一些选项参数,grpc_retry 就是经过这个机制实现的,能够实现自动重试功能
转载请注明出处
本文连接: http://hatlonely.github.io/20...