本文主要介绍了Go语言中函数式选项模式及该设计模式在实际编程中的应用。git
最近看go-micro/options.go源码的时候,发现了一段关于服务注册的代码以下:github
type Options struct { Broker broker.Broker Cmd cmd.Cmd Client client.Client Server server.Server Registry registry.Registry Transport transport.Transport // Before and After funcs BeforeStart []func() error BeforeStop []func() error AfterStart []func() error AfterStop []func() error // Other options for implementations of the interface // can be stored in a context Context context.Context } func newOptions(opts ...Option) Options { opt := Options{ Broker: broker.DefaultBroker, Cmd: cmd.DefaultCmd, Client: client.DefaultClient, Server: server.DefaultServer, Registry: registry.DefaultRegistry, Transport: transport.DefaultTransport, Context: context.Background(), } for _, o := range opts { o(&opt) } return opt }
当时呢,也不是很明白newOptions
这个构造函数为何要这么写,可是后面在微信群里看到有人也再发相似的代码问为何要这么写,后来在群里讨论的时候才知道了这是一种设计模式–函数式选项模式。golang
可能你们看到如今也不是很明白我说的问题究竟是什么,我把它简单提炼一下。编程
咱们如今有一个结构体,定义以下:设计模式
type Option struct { A string B string C int }
如今咱们须要为其编写一个构造函数,咱们可能会写成下面这种方式:bash
func newOption(a, b string, c int) *Option { return &Option{ A: a, B: b, C: c, } }
上面的代码很好理解,也是咱们一直在写的。有什么问题吗?微信
咱们如今来思考如下两个问题:闭包
咱们先定义一个OptionFunc
的函数类型函数
type OptionFunc func(*Option)
而后利用闭包为每一个字段编写一个设置值的With函数:测试
func WithA(a string) OptionFunc { return func(o *Option) { o.A = a } } func WithB(b string) OptionFunc { return func(o *Option) { o.B = b } } func WithC(c int) OptionFunc { return func(o *Option) { o.C = c } }
而后,咱们定义一个默认的Option
以下:
var ( defaultOption = &Option{ A: "A", B: "B", C: 100, } )
最后编写咱们新版的构造函数以下:
func newOption2(opts ...OptionFunc) (opt *Option) { opt = defaultOption for _, o := range opts { o(opt) } return }
测试一下:
func main() { x := newOption("nazha", "小王子", 10) fmt.Println(x) x = newOption2() fmt.Println(x) x = newOption2( WithA("沙河娜扎"), WithC(250), ) fmt.Println(x) }
输出:
&{nazha 小王子 10} &{A B 100} &{沙河娜扎 B 250}
这样一个使用函数式选项设计模式的构造函数就实现了。这样默认值也有了,之后再要为Option添加新的字段也不会影响以前的代码。
推荐阅读: