Katakan kita mentakrifkan objek struktur pengguna Pengguna dalam kod, yang mempunyai sifat berikut.
type User struct { ID string // 必需项 Name string // 必需项 Age int // 非必需项 Gender bool // 非必需项 }
Apabila memulakan objek, cara paling mudah ialah mengisi terus nilai atribut, seperti
u := &User{ID: "12glkui234d", Name: "菜刀", Age: 18, Gender: true}
Tetapi terdapat masalah: atribut dalam objek Pengguna tidak semestinya boleh dieksport mempunyai medan atribut ialah kata laluan (huruf pertama adalah huruf kecil, bukan dieksport. Jika objek Pengguna perlu dibina dalam modul lain, medan kata laluan tidak boleh diisi).
Jadi kita perlu mentakrifkan fungsi untuk membina objek Pengguna Kaedah pembina paling mudah yang boleh difikirkan adalah seperti berikut.
func NewUser(id, name string, age int, gender bool) *User { return &User{ ID: id, Name: name, Age: age, Gender: gender, } }
Tetapi terdapat juga beberapa masalah: untuk objek Pengguna, hanya atribut ID dan Nama diperlukan, Umur dan Jantina adalah pilihan, dan nilai lalai tidak boleh ditetapkan Sebagai contoh, nilai lalai Umur ialah 0 . Nilai lalai Jantina adalah palsu, yang jelas tidak munasabah.
Menghadapi masalah ini, apakah penyelesaian yang boleh kita pakai?
Penyelesaian paling kasar yang boleh kita fikirkan ialah menetapkan pembina untuk setiap situasi parameter. Seperti yang ditunjukkan dalam kod berikut
func NewUser(id, name string) *User { return &User{ID: id, Name: name} } func NewUserWithAge(id, name string, age int) *User { return &User{ID: id, Name: name, Age: age} } func NewUserWithGender(id, name string, gender bool) *User { return &User{ID: id, Name: name, Gender: gender} } func NewUserWithAgeGender(id, name string, age int, gender bool) *User { return &User{ID: id, Name: name, Age: age, Gender: gender} }
这种方式适合参数较少且不易发生变化的情况。该方式在 Go 标准库中也有使用,例如 net 包中的 Dial 和 DialTimeout 方法。
func Dial(network, address string) (Conn, error) {} func DialTimeout(network, address string, timeout time.Duration) (Conn, error) {}
但该方式的缺陷也很明显:试想,如果构造对象 User 增加了参数字段 Phone,那么我们需要新增多少个组合函数?
另外一种常见的方式是配置化,我们将所有可选的参数放入一个 Config 的配置结构体中。
type User struct { ID string Name string Cfg *Config } type Config struct { Age int Gender bool } func NewUser(id, name string, cfg *Config) *User { return &User{ID: id, Name: name, Cfg: cfg} }
这样,我们只需要一个 NewUser() 函数,不管之后增加多少配置选项,NewUser 函数都不会得到破坏。
但是,这种方式,我们需要先构造 Config 对象,这时候对 Config 的构造又回到了方案一中存在的问题。
面对这样的问题,我们还可以选择函数式选项模式。
首先,我们定义一个 Option 函数类型
type Option func(*User)
然后,为每个属性值定义一个返回 Option 函数的函数
func WithAge(age int) Option { return func(u *User) { u.Age = age } } func WithGender(gender bool) Option { return func(u *User) { u.Gender = gender } }
此时,我们将 User 对象的构造函数改为如下所示
func NewUser(id, name string, options ...Option) *User { u := &User{ID: id, Name: name} for _, option := range options { option(u) } return u }
按照这种构造方式,我们就可以这样配置 User 对象了
u := NewUser("12glkui234d", "菜刀", WithAge(18), WithGender(true))
以后不管 User 增加任何参数 XXX,我们只需要增加对应的 WithXXX 函数即可,是不是非常地优雅?
Functional Options 这种编程模式,我们经常能在各种项目中找到它的身影。例如,我在 tidb 项目中仅使用 opts ... 关键字搜索,就能看到这么多使用了 Functional Options 的代码(截图还未包括全部)。
Corak pilihan berfungsi menyelesaikan masalah cara mengkonfigurasi parameter untuk objek secara dinamik dan fleksibel, tetapi ia perlu digunakan dalam senario yang sesuai.
Apabila parameter konfigurasi objek adalah kompleks, seperti banyak parameter pilihan, medan tidak diimport, parameter mungkin meningkat dengan versi, dll., maka mod pilihan berfungsi boleh membantu kami dengan baik.
Atas ialah kandungan terperinci Model pengaturcaraan yang aktif dalam banyak projek Go. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!