Go泛型應(yīng)用工廠方法及泛型使用
前言
由于網(wǎng)上關(guān)于泛型使用的文章太多了,這里就不聊怎么使用泛型了,今天我們結(jié)合工廠方法+泛型方法來(lái)看一下泛型到底是如何在業(yè)務(wù)場(chǎng)景中使用的。本文涉及到的點(diǎn)如下:
- 接口是怎么實(shí)現(xiàn)泛化編程的。
- 泛型是怎么解決接口的局限性的。
- 泛型使用的最佳時(shí)機(jī)。
- 關(guān)于功能設(shè)計(jì)的簡(jiǎn)單建議。
話不多說(shuō),我們開(kāi)始吧,具體泛型怎么使用及語(yǔ)法,請(qǐng)自行查閱相關(guān)資料。
接口實(shí)現(xiàn)泛化編程
平時(shí)我們編寫(xiě)結(jié)構(gòu)體和方法的時(shí)候,一般是使用具體的類型:要么是基本類型,要么是自定義類型。但是如果要編寫(xiě)可以應(yīng)用于多種類型的代碼的時(shí)候,那么這種限制對(duì)程序的束縛就會(huì)比較大。
那么我們想編出一些泛化的方法和接口,怎么辦呢? 這個(gè)時(shí)候我們想到了接口,如果方法的參數(shù)是一個(gè)接口,而不是一個(gè)結(jié)構(gòu)體,這樣對(duì)程序的限制就會(huì)放開(kāi)了許多。因?yàn)槿魏螌?shí)現(xiàn)該接口的結(jié)構(gòu)體都可以作為該方法的接口參數(shù),這樣就可以保證后面在添加其他同類功能的時(shí)候,只需實(shí)現(xiàn)這個(gè)接口就可以滿足需求了。
比如如下一個(gè)需求:
定義兩個(gè)手機(jī)品牌結(jié)構(gòu)體華為,蘋(píng)果 ,并打印出各自品牌的名字。保證程序的擴(kuò)展性,后面我們可能還要加入小米
import "fmt"
//手機(jī)統(tǒng)一接口
type Phone interface {
PrintBrand()
}
?
type HuaweiPhone struct {
}
func (hw *HuaweiPhone) PrintBrand() {
fmt.Printf("品牌名字:華為")
}
?
type Iphone struct {
}
func (ip *Iphone) PrintBrand() {
fmt.Printf("品牌名字:蘋(píng)果")
}
//統(tǒng)一打印方法
func PrintBrand(phone Phone) {
phone.PrintBrand()
}
func main() {
hw := HuaweiPhone{}
ip := Iphone{}
PrintBrand(ip)
PrintBrand(hw)
}如上面代碼,我們定義了兩個(gè)手機(jī)品牌的結(jié)構(gòu)體,我們想打印各個(gè)手機(jī)的品牌名字,需要調(diào)用統(tǒng)一打印方法就可以了,如果后面添加其他品牌的話,我們只需要實(shí)現(xiàn)Phone這個(gè)接口就可以,如下添加小米手機(jī)品牌:
type XiaomiPhone struct {
}
func (xm XiaomiPhone) PrintBrand() {
fmt.Printf("品牌名字:小米")
}以上代碼我們可以看到,通過(guò)接口也可以定義一些泛化的行為。
工廠+泛型來(lái)實(shí)現(xiàn)更通用的泛化編程
可是有時(shí)候,即便我們使用了接口,對(duì)程序的約束依然還是很強(qiáng),因?yàn)橐坏┪覀冎该髁司唧w的接口,就會(huì)要求我們必須使用特定的接口。而我們希望編寫(xiě)更通用的代碼,要使代碼能夠應(yīng)用于某種不具體的類型,而不是具體的一個(gè)接口或者結(jié)構(gòu)體。這個(gè)要怎么辦呢?
比如,我們基于以上的需求繼續(xù)加需求
由于我們對(duì)接的品牌增大到20種,除了上面三種還有 魅族、三星、諾基亞、中興。。。。等等
這個(gè)時(shí)候我們基于當(dāng)前的代碼已經(jīng)不能滿足,那么我們想到了工廠設(shè)計(jì)模式,在工廠中用泛型來(lái)泛化所有的類型,我們通過(guò)傳入類型名字來(lái)打印出具體的品牌名。我們引入工廠模式繼續(xù)優(yōu)化我們的代碼,
如下:
var cache sync.Map
//工廠方法 可以傳入任意的類型
func PhoneFactory[T any]() (t *T) {
target := reflect.TypeOf(t)
v, ok := cache.Load(t)
if ok {
return v.(*T)
}
v = new(T)
v, _ = cache.LoadOrStore(target, v)
return v.(*T)
}
func main() {
PrintBrand(PhoneFactory[Iphone]())
PrintBrand(PhoneFactory[HuaweiPhone]())
PrintBrand(PhoneFactory[XiaomiPhone]())
}代碼中我們編寫(xiě)了個(gè)工廠方法,泛型類型為 any, 接收任意的類型,在工廠中我們創(chuàng)建對(duì)象返回相應(yīng)的類型并緩存類型對(duì)象防止重復(fù)創(chuàng)建。這樣我們后面再加其他類別的時(shí)候可以通過(guò)這個(gè)工廠方法來(lái)統(tǒng)一的創(chuàng)建,我們還可以通過(guò)反射在創(chuàng)建前后根據(jù)業(yè)務(wù)需要做一些操作。
泛型使用的最佳時(shí)機(jī)
泛型的加入,無(wú)疑增加了代碼的復(fù)雜度,那么我們使用泛型的最佳時(shí)機(jī)是什么時(shí)候呢?
Go 泛型主要設(shè)計(jì)者 Ian Lance Taylor 給出了簡(jiǎn)要的泛型使用方針,當(dāng)開(kāi)發(fā)者發(fā)現(xiàn)自己多次編寫(xiě)完全相同的代碼,而這些副本之間的唯一區(qū)別僅在于使用了不同類型,這時(shí)候便可以考慮使用類型參數(shù)。換句話說(shuō),即開(kāi)發(fā)者應(yīng)避免使用類型參數(shù),直到發(fā)現(xiàn)自己要多次編寫(xiě)完全相同的代碼。
關(guān)于功能設(shè)計(jì)的簡(jiǎn)單建議
比如說(shuō)上面的業(yè)務(wù),其實(shí)我們開(kāi)始設(shè)計(jì)的時(shí)候設(shè)計(jì)到接口層面就可以了,如果一開(kāi)始就引入工廠方法,其實(shí)這算是過(guò)度設(shè)計(jì),我們?cè)O(shè)計(jì)一個(gè)功能的原則是,抓住上下文,適度設(shè)計(jì),因?yàn)橐坏┪覀兺度肓诉^(guò)多的精力到靈活設(shè)計(jì)上,勢(shì)必會(huì)影響本應(yīng)該完成的需求。同時(shí),過(guò)多的功能會(huì)引入更多潛在的問(wèn)題,而修復(fù)問(wèn)題也會(huì)耗費(fèi)我們的時(shí)間和精力。而且在當(dāng)前這個(gè)敏捷開(kāi)發(fā)的時(shí)代,更是如此。
最后
為了提高可閱讀性,以上代碼都是以最簡(jiǎn)單的方式呈現(xiàn)的,實(shí)際業(yè)務(wù)遠(yuǎn)比這要復(fù)雜的多,這里只是提供一種方向。
到此這篇關(guān)于Go泛型應(yīng)用工廠方法及泛型使用的文章就介紹到這了,更多相關(guān)Go泛型應(yīng)用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang實(shí)現(xiàn)簡(jiǎn)單rpc調(diào)用過(guò)程解析
這篇文章主要介紹了golang實(shí)現(xiàn)簡(jiǎn)單rpc調(diào)用,包括RPC具體實(shí)現(xiàn)結(jié)合實(shí)例代碼給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-05-05
GoLang中sql.Exec()報(bào)錯(cuò)解決辦法
這篇文章主要給大家介紹了關(guān)于GoLang中sql.Exec()報(bào)錯(cuò)的解決辦法,文中通過(guò)代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2024-01-01
gRPC的發(fā)布訂閱模式及REST接口和超時(shí)控制
這篇文章主要為大家介紹了gRPC的發(fā)布訂閱模式及REST接口和超時(shí)控制,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06
探索分析Go?HTTP?GET請(qǐng)求發(fā)送body
這篇文章主要為大家介紹了探索分析Go?HTTP?GET請(qǐng)求發(fā)送body,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
golang快速實(shí)現(xiàn)網(wǎng)頁(yè)截圖的方法
這篇文章主要介紹了golang快速實(shí)現(xiàn)網(wǎng)頁(yè)截圖的方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03
golang Iris運(yùn)行多個(gè)應(yīng)用的實(shí)現(xiàn)
本文主要介紹了golang Iris運(yùn)行多個(gè)應(yīng)用的實(shí)現(xiàn),在Iris里面,提供了一種方式可以讓我們同時(shí)運(yùn)行多個(gè)應(yīng)用,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01
Go 并發(fā)實(shí)現(xiàn)協(xié)程同步的多種解決方法
這篇文章主要介紹了Go 并發(fā)——實(shí)現(xiàn)協(xié)程同步的多種解決方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-08-08
Golang中文件目錄操作的實(shí)現(xiàn)步驟詳解
在Golang中,文件目錄是指計(jì)算機(jī)文件系統(tǒng)中的文件夾或目錄。目錄是用于組織和存儲(chǔ)文件的一種方式,可以包含文件和其他子目錄,本文主要介紹了Golang中文件目錄操作的實(shí)現(xiàn)方法,需要的朋友可以參考下2023-05-05

