Go編程中常見(jiàn)錯(cuò)誤和不良實(shí)踐解析
Go常見(jiàn)錯(cuò)誤和不良實(shí)踐
Go編程的某些實(shí)踐容易被誤用或忽視,了解這些特性的特點(diǎn)和陷阱,可以幫助我們編寫(xiě)更好的代碼。原文: 5+ BAD Practices In Go: Learn From Mistakes[1]
使用Go和使用其他編程語(yǔ)言中一樣,需要了解常見(jiàn)錯(cuò)誤和不良實(shí)踐,才能編寫(xiě)既干凈又高效的代碼。
本文討論的一些實(shí)踐并不一定都是不好的,在特定情況下很有用。 然而,我們需要知道可能會(huì)有什么問(wèn)題,為什么應(yīng)該回避某些習(xí)慣,以及如何避開(kāi)常見(jiàn)的陷阱。
1. 使用init()
Go中的init()
函數(shù)是一個(gè)特殊函數(shù),在main函數(shù)之前執(zhí)行。
"如果初始化對(duì)于任何包都很重要,為什么init()在Go中被認(rèn)為是一個(gè)不好的做法?"
是的,雖然init()
函數(shù)確實(shí)有助于在運(yùn)行核心邏輯之前進(jìn)行初始化,但其執(zhí)行順序可能很難理解,可能導(dǎo)致對(duì)初始化順序的混淆。
// package A func init() {} // package B func init() {} // which run first?
想象一下,有兩個(gè)模塊在安裝時(shí)相互依賴,但位于不同的包中。結(jié)果我們最終需要編寫(xiě)更復(fù)雜的代碼來(lái)管理時(shí)序,更糟的是,甚至可能陷入死鎖情況。
使用init()
的另一個(gè)缺點(diǎn)是測(cè)試會(huì)變得復(fù)雜。因?yàn)檫@些函數(shù)是自動(dòng)運(yùn)行的,無(wú)法選擇何時(shí)執(zhí)行。
缺乏控制使得設(shè)置測(cè)試用例成為一項(xiàng)挑戰(zhàn)。
我曾經(jīng)遇到過(guò)一個(gè)問(wèn)題,我的服務(wù)在部署后花了很長(zhǎng)時(shí)間才準(zhǔn)備好。我在main()
函數(shù)的開(kāi)始處設(shè)置了一個(gè)斷點(diǎn),但從未觸發(fā)。
經(jīng)過(guò)冗長(zhǎng)的調(diào)試后,我們發(fā)現(xiàn)一個(gè)成員使用了某個(gè)包中的init()
函數(shù)從一個(gè)大文件加載一個(gè)大數(shù)據(jù)集,這讓我們花費(fèi)大量時(shí)間去解決這么一個(gè)小問(wèn)題。
2. 使用全局變量
Go中的全局變量可能會(huì)帶來(lái)類似單例的問(wèn)題,特別是當(dāng)這些全局變量很復(fù)雜時(shí)(比如映射、切片或指針)。
"那么,全局變量有什么大不了的?"
競(jìng)爭(zhēng)條件: 當(dāng)有多個(gè)程序試圖同時(shí)訪問(wèn)同一個(gè)全局變量時(shí),事情可能會(huì)變得混亂。
更少的可測(cè)試性: 應(yīng)用程序依賴于全局變量,意味著有狀態(tài),從而在單元或集成測(cè)試期間,這些全局變量需要與main()
函數(shù)中的內(nèi)容或在生產(chǎn)環(huán)境中部署的內(nèi)容保持一致。
模塊化程度較低,可重用性較差: 可以從任何地方訪問(wèn)全局變量,很難跟蹤其使用方式和位置。
因此,這里的建議是保持對(duì)包的封裝。
從而使得代碼更容易移動(dòng),并且不太可能破壞其他東西。通過(guò)避免使用全局變量,可以使代碼不那么受約束,并且更容易更新或復(fù)用。
3. 忽略錯(cuò)誤信息
用Go編程時(shí),錯(cuò)誤是不可避免的,知道如何處理錯(cuò)誤可以讓我們避免各種各樣的問(wèn)題。
"忽略錯(cuò)誤真的那么糟糕嗎?"
是的,完全正確。
一些Go新手可能會(huì)用"_"符號(hào)將錯(cuò)誤撇在一邊,但忽略函數(shù)返回的錯(cuò)誤值,可能會(huì)帶來(lái)麻煩。
如果不對(duì)錯(cuò)誤進(jìn)行管理,也許程序會(huì)出現(xiàn)panic和crash。
// sample 1 func main() { var x interface{} = "hello" s := x.(int) // panic: interface conversion: interface {} is string, not int fmt.Println(s) } // sample 2 func main() { var x interface{} = "hello" s, _ := x.(int) // safe but DON'T fmt.Println(s) }
跳過(guò)錯(cuò)誤可能會(huì)適得其反,尤其是對(duì)于線上生產(chǎn)環(huán)境,調(diào)試會(huì)成為一場(chǎng)噩夢(mèng)。總是--我的意思是總是--檢查錯(cuò)誤并采取正確的措施以保持代碼順利運(yùn)行。
4. 避免GOTO
無(wú)論用Go還是其他語(yǔ)言,避免使用"goto"是大家的共識(shí)。
使用goto會(huì)破壞代碼的自然流程。
會(huì)破壞我們理解不同代碼段之間關(guān)系的方式,讓我們很難在不弄得亂七八糟的情況下修改代碼。
此外,調(diào)試也變得更加令人困惑,測(cè)試也更加棘手。
從本質(zhì)上講,依賴goto往往會(huì)產(chǎn)生更多錯(cuò)誤,并難以深入了解問(wèn)題。因此,作為最佳實(shí)踐,明智的做法是避開(kāi)它。
5. 跳過(guò)Defer和Recover
如果你忽略"defer"和"recover",就失去了對(duì)panic的堅(jiān)實(shí)保護(hù)。
為什么?
因?yàn)楫?dāng)出現(xiàn)panic時(shí),"defer"仍然會(huì)起作用,而"recover"會(huì)抓住panic,讓我們有機(jī)會(huì)處理不可預(yù)見(jiàn)的問(wèn)題[2]。
看看這個(gè)例子,其中'file.Close()'只是放在末尾,這不是一個(gè)Go風(fēng)格的解決方案:
func readFile(filename string) { file, err := os.Open(filename) if err != nil { panic(err) } // Do something with the file file.Close() // <--- DONT }
相反,像這樣使用"defer":
func readFile(filename string) { file, err := os.Open(filename) if err != nil { panic(err) } defer file.Close() // Do something with the file ... }
在打開(kāi)文件后立即調(diào)用defer file.Close()
可以確保即使readFile()
遇到panic,文件也會(huì)被關(guān)閉。此外,還可以方便的提醒我們?cè)诖蜷_(kāi)資源后立即進(jìn)行清理。
6. 過(guò)多使用context.Background()
Go的context
功能非常有用,當(dāng)代碼與數(shù)據(jù)庫(kù)或網(wǎng)站對(duì)話時(shí),有助于管理時(shí)間限制等事情。
如果沒(méi)有設(shè)定截止時(shí)間,應(yīng)用可能會(huì)陷入阻塞,被數(shù)以百萬(wàn)計(jì)的請(qǐng)求淹沒(méi)。
通過(guò)一個(gè)特殊功能,可以很容易的設(shè)置時(shí)間限制。
該函數(shù)有三種時(shí)間選擇: Fast(0.5秒)、Medium(3秒)和Slow(10秒)。這樣就不用一直使用context.Background(),而且可以為每個(gè)任務(wù)選擇合適的時(shí)間限制。
以下是Fast的一些示例代碼:
const FastTimeout = 500 * time.Millisecond func WrapCustomContext(ctx context.Context, dur time.Duration) (context.Context, context.CancelFunc) { return context.WithTimeout(ctx, dur) } func GenFastContext() (context.Context, context.CancelFunc) { return WrapCustomContext(context.Background(), FastTimeout) } func WrapFastContext(ctx context.Context) (context.Context, context.CancelFunc) { return WrapCustomContext(ctx, FastTimeout) }
有了這些函數(shù),就可以選擇正確的時(shí)間限制,應(yīng)用也因此運(yùn)行得更好。
好還是不好,只是一些概念,我們可以決定其真正含義。
所以,明智的使用"不好"的特性,它就能變成"最好"的方案。
參考資料
[1]
5+ BAD Practices In Go: Learn From Mistakes:
https://levelup.gitconnected.com/5-bad-practices-in-go-learn-from-mistakes-13afb4d303b3
[2]
What you know about defer in Go is not enough!:
https://medium.com/@func25/what-you-know-about-defer-in-go-is-not-enough-2681d4b128c3
以上就是Go編程中常見(jiàn)錯(cuò)誤和不良實(shí)踐解析的詳細(xì)內(nèi)容,更多關(guān)于Go錯(cuò)誤解析的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- go高并發(fā)時(shí)append方法偶現(xiàn)錯(cuò)誤解決分析
- 詳解Go多協(xié)程并發(fā)環(huán)境下的錯(cuò)誤處理
- Golang錯(cuò)誤處理:異常捕捉和恢復(fù)機(jī)制
- Go語(yǔ)言常見(jiàn)錯(cuò)誤之a(chǎn)ny沒(méi)傳遞任何信息解決分析
- Go語(yǔ)言常見(jiàn)錯(cuò)誤之將接口定義在實(shí)現(xiàn)方
- Go語(yǔ)言常見(jiàn)錯(cuò)誤之誤用init函數(shù)實(shí)例解析
- Go語(yǔ)言常見(jiàn)錯(cuò)誤接口污染解決分析
- Go語(yǔ)言常見(jiàn)錯(cuò)誤之濫用getters/setters誤區(qū)實(shí)例探究
- Go并發(fā)編程中的錯(cuò)誤恢復(fù)機(jī)制與代碼持續(xù)執(zhí)行實(shí)例探索
相關(guān)文章
Go語(yǔ)言中一些不常見(jiàn)的命令參數(shù)詳解
這篇文章主要給大家介紹了關(guān)于Go語(yǔ)言中一些不常見(jiàn)的命令參數(shù)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-12-12GoLang?channel關(guān)閉狀態(tài)相關(guān)操作詳解
Channel?和?goroutine?的結(jié)合是?Go?并發(fā)編程的大殺器。而?Channel?的實(shí)際應(yīng)用也經(jīng)常讓人眼前一亮,通過(guò)與?select,cancel,timer?等結(jié)合,它能實(shí)現(xiàn)各種各樣的功能。接下來(lái),我們就要介紹GoLang?channel關(guān)閉狀態(tài)相關(guān)操作2022-10-10GO語(yǔ)言實(shí)現(xiàn)AES-CFB加密的操作方法
本文介紹了如何用Go語(yǔ)言實(shí)現(xiàn)AES-CFB加密和解密,首先,定義一個(gè)屬于encrypt包的文件,使用AES算法、CFB模式和Base64編碼等功能,在加密函數(shù)中,接收明文和密鑰,生成一個(gè)AES塊密碼和一個(gè)隨機(jī)的初始化向量,實(shí)現(xiàn)明文的加密2024-10-10