網站首頁 編程語言 正文
前言
由于網上關于泛型使用的文章太多了,這里就不聊怎么使用泛型了,今天我們結合工廠方法+泛型方法來看一下泛型到底是如何在業務場景中使用的。本文涉及到的點如下:
- 接口是怎么實現泛化編程的。
- 泛型是怎么解決接口的局限性的。
- 泛型使用的最佳時機。
- 關于功能設計的簡單建議。
話不多說,我們開始吧,具體泛型怎么使用及語法,請自行查閱相關資料。
接口實現泛化編程
平時我們編寫結構體和方法的時候,一般是使用具體的類型:要么是基本類型,要么是自定義類型。但是如果要編寫可以應用于多種類型的代碼的時候,那么這種限制對程序的束縛就會比較大。
那么我們想編出一些泛化的方法和接口,怎么辦呢??這個時候我們想到了接口,如果方法的參數是一個接口,而不是一個結構體,這樣對程序的限制就會放開了許多。因為任何實現該接口的結構體都可以作為該方法的接口參數,這樣就可以保證后面在添加其他同類功能的時候,只需實現這個接口就可以滿足需求了。
比如如下一個需求:
定義兩個手機品牌結構體華為,蘋果 ,并打印出各自品牌的名字。保證程序的擴展性,后面我們可能還要加入小米
import "fmt" //手機統一接口 type Phone interface { PrintBrand() } ? type HuaweiPhone struct { } func (hw *HuaweiPhone) PrintBrand() { fmt.Printf("品牌名字:華為") } ? type Iphone struct { } func (ip *Iphone) PrintBrand() { fmt.Printf("品牌名字:蘋果") } //統一打印方法 func PrintBrand(phone Phone) { phone.PrintBrand() } func main() { hw := HuaweiPhone{} ip := Iphone{} PrintBrand(ip) PrintBrand(hw) }
如上面代碼,我們定義了兩個手機品牌的結構體,我們想打印各個手機的品牌名字,需要調用統一打印方法就可以了,如果后面添加其他品牌的話,我們只需要實現Phone
這個接口就可以,如下添加小米手機品牌:
type XiaomiPhone struct { } func (xm XiaomiPhone) PrintBrand() { fmt.Printf("品牌名字:小米") }
以上代碼我們可以看到,通過接口也可以定義一些泛化的行為。
工廠+泛型來實現更通用的泛化編程
可是有時候,即便我們使用了接口,對程序的約束依然還是很強,因為一旦我們指明了具體的接口,就會要求我們必須使用特定的接口。而我們希望編寫更通用的代碼,要使代碼能夠應用于某種不具體的類型
,而不是具體的一個接口或者結構體。這個要怎么辦呢?
比如,我們基于以上的需求繼續加需求
由于我們對接的品牌增大到20種,除了上面三種還有 魅族、三星、諾基亞、中興。。。。等等
這個時候我們基于當前的代碼已經不能滿足,那么我們想到了工廠設計模式,在工廠中用泛型來泛化所有的類型,我們通過傳入類型名字來打印出具體的品牌名。我們引入工廠模式繼續優化我們的代碼,
如下:
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]()) }
代碼中我們編寫了個工廠方法,泛型類型為 any, 接收任意的類型,在工廠中我們創建對象返回相應的類型并緩存類型對象防止重復創建。這樣我們后面再加其他類別的時候可以通過這個工廠方法來統一的創建,我們還可以通過反射在創建前后根據業務需要做一些操作。
泛型使用的最佳時機
泛型的加入,無疑增加了代碼的復雜度,那么我們使用泛型的最佳時機是什么時候呢?
Go 泛型主要設計者 Ian Lance Taylor 給出了簡要的泛型使用方針,當開發者發現自己多次編寫完全相同的代碼,而這些副本之間的唯一區別僅在于使用了不同類型,這時候便可以考慮使用類型參數。換句話說,即開發者應避免使用類型參數,直到發現自己要多次編寫完全相同的代碼。
關于功能設計的簡單建議
比如說上面的業務,其實我們開始設計的時候設計到接口層面就可以了,如果一開始就引入工廠方法,其實這算是過度設計,我們設計一個功能的原則是,抓住上下文,適度設計,因為一旦我們投入了過多的精力到靈活設計上,勢必會影響本應該完成的需求。同時,過多的功能會引入更多潛在的問題,而修復問題也會耗費我們的時間和精力。而且在當前這個敏捷開發的時代,更是如此。
最后
為了提高可閱讀性,以上代碼都是以最簡單的方式呈現的,實際業務遠比這要復雜的多,這里只是提供一種方向。
原文鏈接:https://juejin.cn/post/7120104935953285134
相關推薦
- 2022-06-24 基于C++實現去除字符串頭尾指定字符功能_C 語言
- 2022-12-01 Python?subprocess庫六個實例快速掌握_python
- 2022-07-04 C#實現中文日歷Calendar_C#教程
- 2022-11-04 深入了解Python中Lambda函數的用法_python
- 2022-11-17 Go語言學習教程之反射的示例詳解_Golang
- 2023-01-02 Python利用socket實現多進程的端口掃描器_python
- 2022-09-27 如何在Python中利用matplotlib.pyplot畫出函數圖詳解_python
- 2021-12-09 C++中的編譯與鏈接_C 語言
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支