成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Go應用中優雅處理Error的幾個技巧

開發 前端
Go語言很強大并且現在也十分流行 — 許多項目都是用Go語言來實現的,如Kubernetes。

 

Go語言很強大并且現在也十分流行 — 許多項目都是用Go語言來實現的,如Kubernetes。Go語言的一個有趣特性是它的多值返回功能提供了一種與其他編程語言不同的錯誤處理方法。Go將error視為具有預定義類型的值,其本身是一個interface類型。然而,編寫大型應用時需要有包含更多上下文信息的error才有價值,而不僅僅是一個值。本文我們將探討如何封裝Go的error類型以在應用程序中帶來更大的價值。

用戶自定義類型

我們將重寫的Go里自帶的 error 類型,首先從一個自定義的錯誤類型開始,該錯誤類型將在程序中識別為 error 類型。因此,我們引入一個封裝了Go的 error 的新自定義Error類型。

  1. type GoError struct { 
  2.    error 

上下文數據

當我們在Go中說error是一個值時,它是字符串值 - 任何實現了 Error() string 函數的類型都可以視作error類型。將字符串值視為error會使跨層的error處理復雜化,因此處理error字符串信息并不是正確的方法。所以我們可以把字符串和錯誤碼解耦:

  1. type GoError struct { 
  2.    error 
  3.    Code    string 

現在對error的處理將基于錯誤碼 Code 字段而不是字符串。讓我們通過上下文數據進一步對錯誤字符串進行解耦,在上下文數據中可以使用 i18n 包進行國際化。

  1. type GoError struct { 
  2.    error 
  3.    Code    string 
  4.    Data    map[string]interface{} 

Data 包含用于構造錯誤字符串的上下文數據。錯誤字符串可以通過數據模板化:

  1. //i18N def 
  2. "InvalidParamValue""Invalid parameter value '{{.actual}}', expected '{{.expected}}' for '{{.name}}'" 

在i18N定義文件中,錯誤碼 Code 將會映射到使用 Data 構建的模板化的錯誤字符串中。

原因(Causes)

error可能發生在任何一層,有必要為每一層提供處理error的選項,并在不丟失原始error值的情況下進一步使用附加的上下文信息對error進行包裝。 GoError 結構體可以用 Causes 進一步封裝,用來保存整個錯誤堆棧。

  1. type GoError struct { 
  2.    error 
  3.    Code    string 
  4.    Data    map[string]interface{} 
  5.    Causes  []error 

如果必須保存多個error數據,則 causes 是一個數組類型,并將其設置為基本 error 類型,以便在程序中包含該原因的第三方錯誤。

組件(Component)

標記層組件將有助于識別error發生在哪一層,并且可以避免不必要的error wrap。例如,如果 service 類型的error組件發生在服務層,則可能不需要wrap error。檢查組件信息將有助于防止暴露給用戶不應該通知的error,比如數據庫error:

  1. type GoError struct { 
  2.    error 
  3.    Code      string 
  4.    Data      map[string]interface{} 
  5.    Causes    []error 
  6.    Component ErrComponent 
  7.  
  8. type ErrComponent string 
  9. const ( 
  10.    ErrService  ErrComponent = "service" 
  11.    ErrRepo     ErrComponent = "repository" 
  12.    ErrLib      ErrComponent = "library" 

響應類型(ResponseType)

添加一個錯誤響應類型這樣可以支持error分類,以便于了解什么錯誤類型。例如,可以根據響應類型(如 NotFound )對error進行分類,像 DbRecordNotFound 、 ResourceNotFound 、 UserNotFound 等等的error都可以歸類為 NotFound error。這在多層應用程序開發過程中非常有用,而且是可選的封裝:

  1. type GoError struct { 
  2.    error 
  3.    Code         string 
  4.    Data         map[string]interface{} 
  5.    Causes       []error 
  6.    Component    ErrComponent 
  7.    ResponseType ResponseErrType 
  8.  
  9. type ResponseErrType string 
  10.  
  11. const ( 
  12.    BadRequest    ResponseErrType = "BadRequest" 
  13.    Forbidden     ResponseErrType = "Forbidden" 
  14.    NotFound      ResponseErrType = "NotFound" 
  15.    AlreadyExists ResponseErrType = "AlreadyExists" 

重試

在少數情況下,出現error會進行重試。retry字段可以通過設置 Retryable 標記來決定是否要進行error重試:

  1. type GoError struct { 
  2.    error 
  3.    Code         string 
  4.    Message      string 
  5.    Data         map[string]interface{} 
  6.    Causes       []error 
  7.    Component    ErrComponent 
  8.    ResponseType ResponseErrType 
  9.    Retryable    bool 

GoError 接口

通過定義一個帶有 GoError 實現的顯式error接口,可以簡化error檢查:、

  1. package goerr 
  2.  
  3. type Error interface { 
  4.    error 
  5.  
  6.    Code() string 
  7.    Message() string 
  8.    Cause() error 
  9.    Causes() []error 
  10.    Data() map[string]interface{} 
  11.    String() string 
  12.    ResponseErrType() ResponseErrType 
  13.    SetResponseType(r ResponseErrType) Error 
  14.    Component() ErrComponent 
  15.    SetComponent(c ErrComponent) Error 
  16.    Retryable() bool 
  17.    SetRetryable() Error 

抽象error

有了上述的封裝方式,更重要的是對error進行抽象,將這些封裝保存在同一地方,并提供error函數的可重用性

  1. func ResourceNotFound(id, kind string, cause error) GoError { 
  2.    data := map[string]interface{}{"kind": kind, "id": id} 
  3.    return GoError{ 
  4.       Code:         "ResourceNotFound"
  5.       Data:         data, 
  6.       Causes:       []error{cause}, 
  7.       Component:    ErrService, 
  8.       ResponseType: NotFound, 
  9.       Retryable:    false
  10.    } 

這個error函數抽象了 ResourceNotFound 這個error,開發者可以使用這個函數來返回error對象而不是每次創建一個新的對象:

  1. //UserService 
  2. user, err := u.repo.FindUser(ctx, userId) 
  3. if err != nil { 
  4.    if err.ResponseType == NotFound { 
  5.       return ResourceNotFound(userUid, "User", err) 
  6.    } 
  7.    return err 

結論

我們演示了如何使用添加上下文數據的自定義Go的error類型,從而使得error在多層應用程序中更有意義。你可以在 這個GitHub Gist [1]看到完整的代碼實現和定義。

責任編輯:張燕妮 來源: 網管叨bi叨
相關推薦

2022-09-07 00:04:37

JavaScript運算符技巧

2021-07-12 15:35:56

JavaScript代碼運算符

2009-06-25 14:05:40

Java應用技巧

2022-07-18 08:08:16

Go?語言技巧

2021-05-07 06:27:29

JavaScript運算符開發

2022-04-06 07:32:41

Java運算符變量

2021-07-05 07:55:11

Goroutine錯誤語言

2024-05-06 12:30:51

Go語言中間件

2023-11-06 08:01:09

Go同步異步

2024-10-16 12:23:55

技巧Spring驗證

2025-03-31 00:29:44

2025-05-22 09:01:28

2022-07-19 08:01:55

函數Go格式化

2024-02-19 08:07:31

Go版本語言

2009-01-05 09:14:17

.NETcatch性能損失

2025-02-05 08:43:40

2019-03-11 09:18:20

Java 8Stream數據結構

2013-06-03 10:02:53

WAF繞過

2014-07-22 09:01:53

SwiftJSON

2022-06-09 10:42:47

GoJSON
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 午夜久久久久久久久久一区二区 | 欧美日韩亚洲一区 | 91久久精| 97人人草 | 亚洲激情综合 | 亚洲福利一区二区 | 影视一区| 亚洲欧美日韩中文字幕一区二区三区 | 久久这里有精品 | 一区二区三区四区免费在线观看 | 亚洲精品电影在线 | 日本欧美在线 | 天天拍天天插 | 91精品国产麻豆 | 日韩一级免费大片 | 成人h视频在线观看 | 欧美激情一区二区三级高清视频 | 久久久久无码国产精品一区 | 在线观看视频91 | 国产精品成人一区二区三区夜夜夜 | 做a视频| 欧美在线一级 | 国产精品视频偷伦精品视频 | 成人在线视频免费播放 | 久久久精品视频一区二区三区 | 中文欧美日韩 | 99久久精品国产毛片 | 欧美日韩一区二区三区四区 | 久久久久久久久国产精品 | 久久国产精品久久 | 婷婷综合网 | 成人精品福利 | 欧美a级成人淫片免费看 | 美女福利视频一区 | 在线国产一区 | www.久久久久久久久 | 性色的免费视频 | 欧美日韩电影一区二区 | 国产高清视频一区二区 | 日韩国产精品一区二区三区 | 成人做爰999 |