溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Go錯(cuò)誤處理的基本規(guī)則有哪些

發(fā)布時(shí)間:2021-12-21 10:33:00 來(lái)源:億速云 閱讀:134 作者:iii 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要講解了“Go錯(cuò)誤處理的基本規(guī)則有哪些”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“Go錯(cuò)誤處理的基本規(guī)則有哪些”吧!

規(guī)則1-不要忽略錯(cuò)誤

遲早你的函數(shù)將返回失敗,你將花費(fèi)大量時(shí)間來(lái)確定原因并恢復(fù)程序。

處理這些錯(cuò)誤。如果您很急或太累-那么休息一下。

package main

import (
"fmt"
"time"
)

func main() {
// 不要忽略這里的錯(cuò)誤
lucky, _ := ifItCanFailItWill()
}

func ifItCanFailItWill() (string, error) {
nowNs := time.Now().Nanosecond()
if nowNs % 2 == 0 {
return "shinny desired value", nil
}

return "", fmt.Errorf("I will fail one day, handle me")
}

規(guī)則2

盡早返回錯(cuò)誤

首先專注于代碼執(zhí)行的“happy path”可能很自然,但是我更喜歡從驗(yàn)證開(kāi)始,并在最后一切都100%完好時(shí)才返回值。

縮放性差:

func nah() (string, error) {
nowNs := time.Now().Nanosecond()
if nowNs % 2 == 0 && isValid() {
return "shinny desired value", nil
}

return "", fmt.Errorf("I will fail one day, handle me")
}

更專業(yè)的:

func earlyReturnRocks() (string, error) {
nowNs := time.Now().Nanosecond()
if nowNs % 2 > 0 {
return "", fmt.Errorf("time dividability must be OCD compliant")
}

if !isValid() {
return "", fmt.Errorf("a different custom, specific, helpful error message")
}

return "shinny desired value", nil
}

優(yōu)點(diǎn)?

  • 更容易閱讀

  • 輕松添加更多驗(yàn)證

  • 嵌套代碼更少(尤其是在循環(huán)中)

  • 明確關(guān)注安全和錯(cuò)誤處理

  • 每個(gè)if條件可能有特定的錯(cuò)誤消息

規(guī)則3

返回值或錯(cuò)誤(但不能同時(shí)包含兩者)

我看到開(kāi)發(fā)人員同時(shí)使用返回值和錯(cuò)誤。這是一個(gè)壞習(xí)慣。避免這樣做。

令人困惑:

func validateToken() (desiredValue string, expiredAt int, err error) {
nowNs := time.Now().Nanosecond()
if nowNs % 2 > 0 {
// THE expiredAt (nowNs) SHOULD NOT BE RETURNED TOGETHER WITH THE ERR
return "", nowNs, fmt.Errorf("token expired")
}

return "shinny desired value", 0, nil
}

不足?

  • 方法簽名不清楚

  • 必須對(duì)方法進(jìn)行反向工程,以了解返回的值以及何時(shí)返回

沒(méi)錯(cuò),有時(shí)您需要返回有關(guān)該錯(cuò)誤的其他信息,在這種情況下,請(qǐng)創(chuàng)建一個(gè)新的專用Error對(duì)象。

更專業(yè)的:

package main

import (
"fmt"
"github.com/davecgh/go-spew/spew"
"time"
)

func main() {
value, err := validateToken()
if err != nil {
spew.Dump(err.Error())
}

spew.Dump(value)
}

// Compatible with error built-in interface.
//
// type error interface {
// Error() string
// }
type TokenExpiredErr struct {
expiredAt int
}

func (e TokenExpiredErr) Error() string {
return fmt.Sprintf("token expired at block %d", e.expiredAt)
}

func validateToken() (desiredValue string, err error) {
nowNs := time.Now().Nanosecond()
if nowNs % 2 > 0 {
return "", TokenExpiredErr{expiredAt: nowNs}
}

return "shinny desired value", nil
}

規(guī)則4

記日志或返回(但不能同時(shí))

當(dāng)你將錯(cuò)誤寫(xiě)入log時(shí),就是在處理它。不要將錯(cuò)誤返回給呼叫者迫使他處理!

Go錯(cuò)誤處理的基本規(guī)則有哪些

為什么?因?yàn)槟幌雰纱位蚨啻螌?duì)同一條消息記錄日志:

package main

import (
"fmt"
"os"
"time"
)

func main() {
// validateToken() is already doing the logging,
// but I didn't reverse engineer the method so I don't know about that
// and now I will unfortunately end up with the same message being logged twice
_, err := validateToken()
if err != nil {
// I have nowhere to return it, SO I RIGHTFULLY LOG IT
// And I will not ignore a possible error writing err
_, err = fmt.Fprint(os.Stderr, fmt.Errorf("validating token failed. %s", err.Error()))
if err != nil {
// Extremely rare, no other choice
panic(err)
}

os.Exit(1)
}
}

type TokenExpiredErr struct {
expiredAt int
}

func (e TokenExpiredErr) Error() string {
return fmt.Sprintf("token expired at block %d", e.expiredAt)
}

func validateToken() (desiredValue string, err error) {
nowNs := time.Now().Nanosecond()
if nowNs % 2 > 0 {
// DO NOT LOG AND RETURN
// DO NOT LOG AND RETURN
// DO NOT LOG AND RETURN
fmt.Printf("token validation failed. token expired at %d", nowNs)
return "", TokenExpiredErr{expiredAt: nowNs}
}

return "shinny desired value", nil
}

即用日志記錄,又返回錯(cuò)誤會(huì)導(dǎo)致混亂輸出:

token validation failed. token expired at 115431493validating token failed. token expired at block 115431493

更專業(yè)的:log或返回錯(cuò)誤,選其一:

validating token failed. token expired at block 599480733

規(guī)則5

在IDE中將if err != nil配置為宏

感謝各位的閱讀,以上就是“Go錯(cuò)誤處理的基本規(guī)則有哪些”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)Go錯(cuò)誤處理的基本規(guī)則有哪些這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

go
AI