日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢(xún)
選擇下列產(chǎn)品馬上在線(xiàn)溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷(xiāo)解決方案
一個(gè)活躍在眾多Go項(xiàng)目中的編程模式

問(wèn)題

假設(shè)我們?cè)诖a中定義了一個(gè)用戶(hù)的結(jié)構(gòu)體對(duì)象 User,它擁有以下屬性。

type User struct {
ID string // 必需項(xiàng)
Name string // 必需項(xiàng)
Age int // 非必需項(xiàng)
Gender bool // 非必需項(xiàng)
}

初始化該對(duì)象時(shí),最簡(jiǎn)單的方式是直接填充屬性值,例如

u := &User{ID: "12glkui234d", Name: "菜刀", Age: 18, Gender: true}

但是這里存在一個(gè)問(wèn)題:User 對(duì)象中的屬性并不一定都是可導(dǎo)出的,例如 User 有一個(gè)屬性字段為 password(首字母小寫(xiě),非導(dǎo)出),如果在其他模塊中需要構(gòu)造 User 對(duì)象,這樣就不能填充該 password 字段了。

所以我們需要定義構(gòu)造 User 對(duì)象的函數(shù),首先能想到最簡(jiǎn)單的構(gòu)造函數(shù)方式如下。

func NewUser(id, name string, age int, gender bool) *User {
return &User{
ID: id,
Name: name,
Age: age,
Gender: gender,
}
}

但是這樣也存在一些問(wèn)題:對(duì)于 User 對(duì)象而言,只有 ID、Name 屬性是必須的,Age 與 Gender 為非必需項(xiàng),且并不能設(shè)置默認(rèn)值,例如 Age 的默認(rèn)值為 0,Gender 的默認(rèn)值是 false ,這顯然不太合理。

面對(duì)該問(wèn)題,我們可以采用的解決方案有哪些呢?

方案一:多函數(shù)構(gòu)造

我們能想到最粗暴地解決方法是:為每種參數(shù)情況設(shè)置一種構(gòu)造函數(shù)。如下代碼所示

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}
}

這種方式適合參數(shù)較少且不易發(fā)生變化的情況。該方式在 Go 標(biāo)準(zhǔn)庫(kù)中也有使用,例如 net 包中的 Dial 和 DialTimeout 方法。

func Dial(network, address string) (Conn, error) {}
func DialTimeout(network, address string, timeout time.Duration) (Conn, error) {}

但該方式的缺陷也很明顯:試想,如果構(gòu)造對(duì)象 User 增加了參數(shù)字段 Phone,那么我們需要新增多少個(gè)組合函數(shù)?

方案二:配置化

另外一種常見(jiàn)的方式是配置化,我們將所有可選的參數(shù)放入一個(gè) Config 的配置結(jié)構(gòu)體中。

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}
}

這樣,我們只需要一個(gè) NewUser() 函數(shù),不管之后增加多少配置選項(xiàng),NewUser 函數(shù)都不會(huì)得到破壞。

但是,這種方式,我們需要先構(gòu)造 Config 對(duì)象,這時(shí)候?qū)?Config 的構(gòu)造又回到了方案一中存在的問(wèn)題。

方案三:函數(shù)式選項(xiàng)模式

面對(duì)這樣的問(wèn)題,我們還可以選擇函數(shù)式選項(xiàng)模式。

首先,我們定義一個(gè) Option 函數(shù)類(lèi)型

type Option func(*User)

然后,為每個(gè)屬性值定義一個(gè)返回 Option 函數(shù)的函數(shù)

func WithAge(age int) Option {
return func(u *User) {
u.Age = age
}
}

func WithGender(gender bool) Option {
return func(u *User) {
u.Gender = gender
}
}

此時(shí),我們將 User 對(duì)象的構(gòu)造函數(shù)改為如下所示

func NewUser(id, name string, options ...Option) *User {
u := &User{ID: id, Name: name}
for _, option := range options {
option(u)
}
return u
}

按照這種構(gòu)造方式,我們就可以這樣配置 User 對(duì)象了

u := NewUser("12glkui234d", "菜刀", WithAge(18), WithGender(true))

以后不管 User 增加任何參數(shù) XXX,我們只需要增加對(duì)應(yīng)的 WithXXX 函數(shù)即可,是不是非常地優(yōu)雅?

Functional Options 這種編程模式,我們經(jīng)常能在各種項(xiàng)目中找到它的身影。例如,我在 tidb 項(xiàng)目中僅使用 opts ... 關(guān)鍵字搜索,就能看到這么多使用了 Functional Options 的代碼(截圖還未包括全部)。

總結(jié)

函數(shù)式選項(xiàng)模式解決了如何動(dòng)態(tài)靈活地為對(duì)象配置參數(shù)的問(wèn)題, 但是需要在合適的場(chǎng)景才使用它。

當(dāng)對(duì)象的配置參數(shù)復(fù)雜,例如可選參數(shù)多、非導(dǎo)入字段、參數(shù)可能隨版本增加等情況,這時(shí)函數(shù)式選項(xiàng)模式就可以很好地幫助到我們。


網(wǎng)站欄目:一個(gè)活躍在眾多Go項(xiàng)目中的編程模式
瀏覽路徑:http://m.5511xx.com/article/copehee.html