從 Kubernetes 學(xué)習(xí) Go 接口封裝
在 Go 項(xiàng)目開發(fā)中,為了提高代碼的可讀性、可維護(hù)性和可測(cè)試性,合理的封裝和抽象至關(guān)重要。本文將介紹幾種常見的封裝策略,包括使用接口隱藏輸入?yún)?shù)細(xì)節(jié)、方便 Mock 測(cè)試的接口抽象、多種底層實(shí)現(xiàn)的接口封裝,以及對(duì)協(xié)程異常處理、WaitGroup 使用和基于信號(hào)量觸發(fā)邏輯的封裝實(shí)踐。通過這些技巧,可以讓各層代碼只關(guān)注自身職責(zé),實(shí)現(xiàn)低耦合、高復(fù)用的設(shè)計(jì)。
使用接口隱藏輸入?yún)?shù)細(xì)節(jié)
當(dāng)一個(gè)方法的輸入?yún)?shù)是結(jié)構(gòu)體時(shí),內(nèi)部調(diào)用會(huì)暴露過多的細(xì)節(jié)。此時(shí),可以將輸入隱式轉(zhuǎn)換為接口,使內(nèi)部實(shí)現(xiàn)僅能看到所需的方法。
type Kubelet struct{}
func (kl *Kubelet) HandlePodAdditions(pods []*Pod) {
for _, pod := range pods {
fmt.Printf("create pods : %s\n", pod.Status)
}
}
func (kl *Kubelet) Run(updates <-chan Pod) {
fmt.Println(" run kubelet")
go kl.syncLoop(updates, kl)
}
func (kl *Kubelet) syncLoop(updates <-chan Pod, handler SyncHandler) {
for {
select {
case pod := <-updates:
handler.HandlePodAdditions([]*Pod{&pod})
}
}
}
type SyncHandler interface {
HandlePodAdditions(pods []*Pod)
}
在這里,Kubelet 本身有多個(gè)方法:
- syncLoop:用于狀態(tài)同步的循環(huán);
- Run:?jiǎn)?dòng)監(jiān)聽循環(huán);
- HandlePodAdditions:處理 Pod 添加的邏輯。
由于 syncLoop 并不需要訪問 kubelet 的其他方法,我們定義了 SyncHandler 接口,讓 kubelet 實(shí)現(xiàn)該接口,并將 kubelet 作為 SyncHandler 傳入 syncLoop,這樣 kubelet 會(huì)被類型轉(zhuǎn)換為 SyncHandler。
轉(zhuǎn)換后,syncLoop 的參數(shù)中將不再暴露 kubelet 的其他方法,使你在編寫 syncLoop 時(shí)更專注于內(nèi)部邏輯。
但這種做法也可能帶來問題:初始抽象可能能滿足第一版需求,但隨著需求增長(zhǎng),如果需要在 syncLoop 中調(diào)用接口未包含的 kubelet 方法,就必須要么顯式傳入 kubelet,要么擴(kuò)展接口,這兩種方式都會(huì)增加編碼成本并破壞原有封裝。
分層封裝與隱藏是設(shè)計(jì)目標(biāo),讓代碼的每一部分只關(guān)注自身職責(zé)。
便于 Mock 測(cè)試的接口封裝
通過接口抽象,我們可以在測(cè)試時(shí)直接實(shí)例化 mock 結(jié)構(gòu)體,用于無需關(guān)注的部分。
type OrderAPI interface {
GetOrderId() string
}
type realOrderImpl struct{}
func (r *realOrderImpl) GetOrderId() string {
return ""
}
type mockOrderImpl struct{}
func (m *mockOrderImpl) GetOrderId() string {
return "mock"
}
這里如果在測(cè)試時(shí)不關(guān)心 GetOrderId 是否正常工作,就可以直接用 mockOrderImpl 初始化 OrderAPI,且 mock 中的邏輯可根據(jù)需要任意復(fù)雜化。
func TestGetOrderId(t *testing.T) {
orderAPI := &mockOrderImpl{} // 如果我們需要獲取訂單 ID,但這不是測(cè)試的重點(diǎn),只需用 mock 結(jié)構(gòu)體初始化
fmt.Println(orderAPI.GetOrderId())
}
gomonkey 也可以用于測(cè)試注入,因此即使現(xiàn)有代碼沒有通過接口封裝,我們?nèi)阅軐?shí)現(xiàn) mock,而且這種方式更為強(qiáng)大。
patches := gomonkey.ApplyFunc(GetOrder, func(orderId string) Order {
return Order{
OrderId: orderId,
OrderState: delivering,
}
})
return func() {
patches.Reset()
}
使用 gomonkey 可以實(shí)現(xiàn)更靈活的 mock,因?yàn)樗梢灾苯釉O(shè)置函數(shù)的返回值,而接口抽象只能處理由結(jié)構(gòu)體實(shí)例化的內(nèi)容。
多種底層實(shí)現(xiàn)的接口封裝
像 iptables 和 ipvs 這樣的實(shí)現(xiàn)是通過接口抽象來完成的,因?yàn)樗芯W(wǎng)絡(luò)設(shè)置都需要同時(shí)處理 Service 和 Endpoint。因此,它們抽象出了 ServiceHandler 和 EndpointSliceHandler:
// ServiceHandler 是用于接收 Service 對(duì)象變更通知的抽象接口
type ServiceHandler interface {
// 當(dāng)檢測(cè)到新的 Service 對(duì)象被創(chuàng)建時(shí)調(diào)用
OnServiceAdd(service *v1.Service)
// 當(dāng)檢測(cè)到已有 Service 對(duì)象被修改時(shí)調(diào)用
OnServiceUpdate(oldService, service *v1.Service)
// 當(dāng)檢測(cè)到已有 Service 對(duì)象被刪除時(shí)調(diào)用
OnServiceDelete(service *v1.Service)
// 當(dāng)所有初始事件處理完成且狀態(tài)已完全同步到本地緩存后調(diào)用
OnServiceSynced()
}
// EndpointSliceHandler 是用于接收 EndpointSlice 對(duì)象變更通知的抽象接口
type EndpointSliceHandler interface {
// 當(dāng)檢測(cè)到新的 EndpointSlice 對(duì)象被創(chuàng)建時(shí)調(diào)用
OnEndpointSliceAdd(endpointSlice *discoveryv1.EndpointSlice)
// 當(dāng)檢測(cè)到已有 EndpointSlice 對(duì)象被修改時(shí)調(diào)用
OnEndpointSliceUpdate(oldEndpointSlice, newEndpointSlice *discoveryv1.EndpointSlice)
// 當(dāng)檢測(cè)到已有 EndpointSlice 對(duì)象被刪除時(shí)調(diào)用
OnEndpointSliceDelete(endpointSlice *discoveryv1.EndpointSlice)
// 當(dāng)所有初始事件處理完成且狀態(tài)已完全同步到本地緩存后調(diào)用
OnEndpointSlicesSynced()
}
然后可以通過 Provider 注入:
type Provider interface {
config.EndpointSliceHandler
config.ServiceHandler
}
這也是我在編寫組件時(shí)最常用的編碼技巧:通過對(duì)相似操作的抽象,上層代碼在替換底層實(shí)現(xiàn)后無需做任何改動(dòng)。
封裝異常處理
如果我們?cè)趩?dòng) goroutine 后不捕獲異常,異常會(huì)導(dǎo)致該 goroutine 直接 panic。但是每次都寫全局的 recover 邏輯并不優(yōu)雅,因此我們可以使用封裝好的 HandleCrash 方法:
package runtime
var (
ReallyCrash = true
)
// 默認(rèn)的全局 Panic 處理器
var PanicHandlers = []func(interface{}){logPanic}
// 支持從外部傳入額外的自定義 panic 處理器
func HandleCrash(additionalHandlers ...func(interface{})) {
if r := recover(); r != nil {
for _, fn := range PanicHandlers {
fn(r)
}
for _, fn := range additionalHandlers {
fn(r)
}
if ReallyCrash {
panic(r)
}
}
}
這既支持內(nèi)部異常處理,也支持外部注入額外的處理器。如果不想讓程序崩潰,可以根據(jù)需要修改邏輯。
package runtime
func Go(fn func()) {
go func() {
defer HandleCrash()
fn()
}()
}
在啟動(dòng) goroutine 時(shí),可以使用 Go 方法,這樣也能避免忘記添加 panic 處理。
封裝 WaitGroup
import "sync"
type Group struct {
wg sync.WaitGroup
}
func (g *Group) Wait() {
g.wg.Wait()
}
func (g *Group) Start(f func()) {
g.wg.Add(1)
go func() {
defer g.wg.Done()
f()
}()
}
這里最重要的是 Start 方法,它在內(nèi)部封裝了 Add 和 Done。雖然只有幾行代碼,但它確保每次使用 WaitGroup 時(shí),都不會(huì)忘記增加或完成計(jì)數(shù)器。
封裝由信號(hào)量觸發(fā)的邏輯
type BoundedFrequencyRunner struct {
sync.Mutex
// Actively triggered
run chan struct{}
// Timer limit
timer *time.Timer
// The actual logic to execute
fn func()
}
func NewBoundedFrequencyRunner(fn func()) *BoundedFrequencyRunner {
return &BoundedFrequencyRunner{
run: make(chan struct{}, 1),
fn: fn,
timer: time.NewTimer(0),
}
}
// Run triggers execution; only one signal can be written here, additional signals are discarded without blocking. You can increase the queue size as needed.
func (b *BoundedFrequencyRunner) Run() {
select {
case b.run <- struct{}{}:
fmt.Println("Signal written successfully")
default:
fmt.Println("Signal already triggered once, discarding")
}
}
func (b *BoundedFrequencyRunner) Loop() {
b.timer.Reset(time.Second * 1)
for {
select {
case <-b.run:
fmt.Println("Run signal triggered")
b.tryRun()
case <-b.timer.C:
fmt.Println("Timer triggered execution")
b.tryRun()
}
}
}
func (b *BoundedFrequencyRunner) tryRun() {
b.Lock()
defer b.Unlock()
// You can add logic here such as rate limiting
b.timer.Reset(time.Second * 1)
b.fn()
}