前段時間准備對線上一個golang系統服務進行內部開源,對代碼里面的錯誤處理進行了一波優化。
優化的幾個原因:
- 錯誤處理信息隨意,未分類未定義。看到錯誤日志不能第一時間定位
- 錯誤的日志重復,有時候一個錯誤經過了好幾層,每一層都會記錄,導致日志混亂
- 錯誤處理不統一,使用不統一,管理也不統一
優化的解決辦法:
- 對錯誤進行分類,統一定義和使用
- 每一個錯誤都有冒泡到包的頂層,處理與記日志。使用方只需定義好自己的信息
實施過程
錯誤分類:函數級,包模塊級,系統api級。
函數級別:
還是采用 err != nil 的形式,並且做一個如下的包裝。
模塊級別
統一返回到對應的goroutine頂層處理
服務對外級別
適當的code和健名的message
底層錯誤級別
考慮及時panic,暴露有用信息
以下為代碼設計:
點擊查看代碼
package ferrors
import (
"fmt"
"golang.org/x/xerrors"
)
//Errors 新的錯誤處理方式
type Errors struct {
Code int64
Msg string
}
// Error 輸出錯誤信息
func (e Errors) Error() string {
return fmt.Sprintf("code: %d msg: %s at: %s", e.Code, e.Msg, "錯誤位置,堆棧信息,可選")
}
// New 創建自定義錯誤
func New(code int64, str string, arg ...interface{}) *Errors {
if len(arg) > 0 {
str = fmt.Sprintf(str, arg...)
}
return &Errors{Code: code, Msg: str}
}
// newErr 創建通用錯誤
func newErr(code int64, err error) *Errors {
switch err := err.(type) {
case *Errors:
return err
case nil:
return &Errors{Code: code, Msg: ""}
default:
return &Errors{Code: code, Msg: err.Error()}
}
}
func NewErrNotFound(err error) error {
return newErr(CodeMkNotFound, err)
}
// ErrorEcho example:使用 error wrapping
func ErrorEcho(err error) string {
return fmt.Sprintf("the error %w", err)
}
//ErrorDump example: xerrors 打印堆棧信息
func ErrorDump() {
err := foo1()
fmt.Printf("%v\n", err)
fmt.Printf("%+v\n", err)
}
var myError = xerrors.New("myerror")
func foo() error {
return myError
}
func foo1() error {
return xerrors.Errorf("foo1 : %w", foo())
}