您好,登錄后才能下訂單哦!
這篇文章主要講解了“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ò)誤返回給呼叫者迫使他處理!
為什么?因?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)注!
免責(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)容。