选项模式(option)
- Option模式的专业术语为:Functional Options Pattern(函数式选项模式)
- Option模式为golang的开发者提供了将一个函数的参数设置为可选的功能,也就是说我们可以选择参数中的某几个,并且可以按任意顺序传入参数。
- 比如针对特殊场景需要不同参数的情况,C++可以直接用重载来写出任意个同名函数,在任意场景调用的时候使用同一个函数名即可;但同样情况下,在golang中我们就必须在不同的场景使用不同的函数,并且参数传递方式可能不同的人写出来是不同的样子,这将导致代码可读性差,维护性差。
Option模式的优缺点
优点
- 支持传递多个参数,并且在参数个数、类型发生变化时保持兼容性
- 任意顺序传递参数
- 支持默认值
- 方便拓展
缺点
- 增加许多function,成本增大
- 参数不太复杂时,尽量少用
代码
package main
//options_ok
import (
"fmt"
)
//使用场景,大量修改一个结构体,增加字段值,初始化极其麻烦
//引入option模式后,直接加一个属性和一个函数就可以搞定
type Message struct {
id int
name string
address string
phone int
}
func (msg Message) String() {
fmt.Printf("ID:%d \n- Name:%s \n- Address:%s \n- phone:%d\n", msg.id, msg.name, msg.address, msg.phone)
}
func New(id, phone int, name, addr string) Message {
return Message{
id: id,
name: name,
address: addr,
phone: phone,
}
}
type Option func(msg *Message)
var DEFAULT_MESSAGE = Message{ id: -1, name: "-1", address: "-1", phone: -1}
func WithID(id int) Option {
return func(m *Message) {
m.id = id
}
}
func WithName(name string) Option {
return func(m *Message) {
m.name = name
}
}
func WithAddress(addr string) Option {
return func(m *Message) {
m.address = addr
}
}
func WithPhone(phone int) Option {
return func(m *Message) {
m.phone = phone
}
}
func NewByOption(opts ...Option) Message {
msg := DEFAULT_MESSAGE
for _, o := range opts {
o(&msg)
}
return msg
}
func NewByOptionWithoutID(id int, opts ...Option) Message {
msg := DEFAULT_MESSAGE
msg.id = id
for _, o := range opts {
o(&msg)
}
return msg
}
func main() {
message1 := New(1, 123, "message1", "cache1")
message1.String()
message2 := NewByOption(
WithID(2),
WithName("message2"),
WithAddress("cache2"),
WithPhone(456),
)
message2.String()
message3 := NewByOptionWithoutID(
3,
WithAddress("cache3"),
WithPhone(789),
WithName("message3"),
)
message3.String()
main1()
}
//例2
type OptionFunc func(op *OptionMenu)
type OptionMenu struct {
op1 string
op2 string
op3 int
op4 int
//......可能还会有跟多的属性加入,这给实例化带来了巨大的问题
}
func InitOptions(optionFuncs ...OptionFunc) OptionMenu {
option := OptionMenu{ }
for _, op := range optionFuncs {
op(&option)
}
return option
}
func WithOp1(op1 string) OptionFunc {
return func(op *OptionMenu) {
op.op1 = op1
}
}
func WithOp2(op2 string) OptionFunc {
return func(op *OptionMenu) {
op.op2 = op2
}
}
func WithOp3(op3 int) OptionFunc {
return func(op *OptionMenu) {
op.op3 = op3
}
}
func WithOp4(op4 int) OptionFunc {
return func(op *OptionMenu) {
op.op4 = op4
}
}
func main1() {
op := InitOptions(
WithOp1("op1"),
WithOp2("op2"),
WithOp3(3),
WithOp4(4),
)
fmt.Printf("%#v\n", op)
}