網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
引言
我看到很多 golang
社區(qū)的開發(fā)者,特別是因?yàn)樗暮?jiǎn)單性而被吸引的開發(fā)者,對(duì) golang
中的事情應(yīng)該如何處理做出了一些快速的判斷。
其中一件事就是錯(cuò)誤處理。由于目前大多數(shù)語(yǔ)言的開發(fā)者都來自于 OOP
背景,他們習(xí)慣于處理異常,或者說"拋出"異常的概念來停止當(dāng)前的應(yīng)用程序的流程,而且他們大多認(rèn)為這也是 golang
的方式,我們必須在出錯(cuò)的情況下停止我們的應(yīng)用程序的流程。他們錯(cuò)了!
不要濫用你的工具
我見過很多,我以前也是這樣做的。每當(dāng)有意外情況發(fā)生時(shí),就用 os.exit(1)
,然后繼續(xù)前進(jìn)。好吧,這不是使用go的正確方法!
我明白為什么這被廣泛使用,因?yàn)榇蠖鄶?shù)早期的 golang
應(yīng)用程序只是終端工具,而且許多這些工具曾經(jīng)使用 .sh
可執(zhí)行文件來構(gòu)建,我們?cè)?jīng)只是退出1;以表示剛剛發(fā)生了一些意外的事情,我們想退出。
我們把這種習(xí)慣帶到了我們的 golang 簡(jiǎn)單的終端應(yīng)用中,然后又帶到了復(fù)雜的應(yīng)用中,這只是另一種 Cargo Cult Programming。 我高度鼓勵(lì)你在不得不這樣做的情況下,要非常小心,因?yàn)樗牵?/p>
- 隨著你的應(yīng)用程序的增長(zhǎng),非常難以維護(hù)。
- 最重要的是,不可能對(duì)這樣的代碼進(jìn)行單元測(cè)試,這顯然表明它的不潔性。
- 以這種方式退出將阻止你的任何延遲操作的執(zhí)行,你的程序?qū)⒘⒓唇K止,這可能導(dǎo)致資源泄漏。請(qǐng)考慮一下這個(gè)例子:
func main() { dbConnection := db.connect("...") defer dbConnection.Close() // this operation won't be executed! entity := Entity{} err := dbConnection.Save(entity) if err != nil { os.Exit(1) } }
考慮傳遞你的錯(cuò)誤
錯(cuò)誤只是 golang
中的另一種類型,你必須用它們來控制程序的執(zhí)行流程。
為了做到這一點(diǎn),我們必須在整個(gè)程序中傳播這些錯(cuò)誤,直到適當(dāng)?shù)奶幚睃c(diǎn)。
考慮一個(gè)管理訂單的HTTP API,我們想禁止客戶在特定條件下下訂單,例如:
package order // package errors var ( UnableToShipToCountry = errors.New("unable to ship order to the requested country") ) type Order struct { // ... order fields } type OrderRepo struct { DB db // ... } func newOrderFromRequest(o OrderRequest) (Order, error) { if o.ShippingAddress.Country != "DE" { return UnableToShipToCountry } // ... the creation logic return Order{...}, nil } func (r *OrderRepo)PlaceOrder(o OrderRequest) error { order, err := newOrderFromRequest(o) if err != nil { // don't handle the error here, its handling may differ return err } // ... db transaction may also return an error return r.db.Save(order) }
在我們的 http package
中:
package http http.HandleFunc("/order", func (w http.ResponseWriter, r *http.Request) { orderRequest := createOrderRequest(r) err := orderRepo.PlaceOrder(orderRequest) if errors.Is(err, order.UnableToShipToCountry) { w.WriteHeader(http.StatusBadRequest) return } if err != nil { // this error in case of DB transaction failure w.WriteHeader(http.StatusInternalServerError) return } // ... w.WriteHeader(http.StatusOK) })
定制你的錯(cuò)誤
我們可以創(chuàng)建我們自己的自定義錯(cuò)誤值,并在我們的程序中使用它,同時(shí)考慮添加一些有用的信息,如錯(cuò)誤跟蹤這可能會(huì)給我們的日志增加一個(gè)有益的價(jià)值,特別是在調(diào)試期間。
type AppErr struct { msg string code int trace string } func (e AppErr) Error() string { return fmt.Sprintf("Msg: %s, code: %d, trace:\n %s", e.msg, e.code, e.trace) } func NewAppErr(msg string, code int) AppErr { stackSlice := make([]byte, 512) s := runtime.Stack(stackSlice, false) return AppErr{msg, code, fmt.Sprintf("\n%s", stackSlice[0:s])} }
而我們?cè)谝粋€(gè)包內(nèi)有這樣一個(gè)用例:
package admin func A() error { return b() } func b() error { return NewAppErr("error from b function!", 3) }
main.go:
func main() { err := admin.A() fmt.Println(err) }
記錄的錯(cuò)誤信息將是:
Msg: error from b function!, code: 3, trace: ?
goroutine 1 [running]: ?
./cmd/app/error.NewAppErr({0x1f42b0, 0x17}, 0x7) ?
./cmd/app/error/error.go:16 +0x35 ?
./cmd/app/admin.b(...) ?
./cmd/app/admin/**admin.go:12** ?
./cmd/app/admin.A(...) ?
./cmd/app/admin/**admin.go:8** ?
main.main() ?
./cmd/app/**main.go:10** +0x8d
你也可以考慮在生產(chǎn)環(huán)境中關(guān)閉你的跟蹤打印,或者通過檢查其他配置值。
原文鏈接:https://juejin.cn/post/7166807817499148302
相關(guān)推薦
- 2022-10-07 C語(yǔ)言一個(gè)函數(shù)如何實(shí)現(xiàn)好幾個(gè)return返回值_C 語(yǔ)言
- 2022-08-30 Mac上出現(xiàn)多個(gè)版本選擇打開edge瀏覽器
- 2023-07-13 el-table實(shí)現(xiàn)多選及反選
- 2022-08-25 python數(shù)學(xué)建模(SciPy+?Numpy+Pandas)_python
- 2022-02-27 select組件選中后獲取當(dāng)前值對(duì)應(yīng)的對(duì)象信息
- 2022-08-27 DOS編寫腳本常用命令整理小結(jié)_DOS/BAT
- 2022-12-16 Docker教程之使用dockerfile生成鏡像_docker
- 2022-04-24 python使用技巧-查找文件?_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支