欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文教你如何封裝安全的go

 更新時間:2022年02月23日 09:15:24   作者:葉劍峰  
這篇文章主要給大家介紹了關(guān)于如何封裝安全go的相關(guān)資料,文中通過實例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下

前言

在業(yè)務(wù)代碼開發(fā)過程中,我們會有很大概率使用go語言的goroutine來開啟一個新的goroutine執(zhí)行另外一段業(yè)務(wù),或者開啟多個goroutine來并行執(zhí)行多個業(yè)務(wù)邏輯。所以我為hade框架增加了兩個方法goroutine.SafeGo 和 goroutine.SafeGoAndWait。

封裝

SafeGo

SafeGo 這個函數(shù),提供了一種goroutine安全的函數(shù)調(diào)用方式。主要適用于業(yè)務(wù)中需要進行開啟異步goroutine業(yè)務(wù)邏輯調(diào)用的場景。

// SafeGo 進行安全的goroutine調(diào)用
// 第一個參數(shù)是context接口,如果還實現(xiàn)了Container接口,且綁定了日志服務(wù),則使用日志服務(wù)
// 第二個參數(shù)是匿名函數(shù)handler, 進行最終的業(yè)務(wù)邏輯
// SafeGo 函數(shù)并不會返回error,panic都會進入hade的日志服務(wù)
func SafeGo(ctx context.Context, handler func())

調(diào)用方式參照如下的單元測試用例:

func TestSafeGo(t *testing.T) {
    container := tests.InitBaseContainer()
    container.Bind(&log.HadeTestingLogProvider{})

    ctx, _ := gin.CreateTestContext(httptest.NewRecorder())
    goroutine.SafeGo(ctx, func() {
        time.Sleep(1 * time.Second)
        return
    })
    t.Log("safe go main start")
    time.Sleep(2 * time.Second)
    t.Log("safe go main end")

    goroutine.SafeGo(ctx, func() {
        time.Sleep(1 * time.Second)
        panic("safe go test panic")
    })
    t.Log("safe go2 main start")
    time.Sleep(2 * time.Second)
    t.Log("safe go2 main end")

}

SafeGoAndWait

SafeGoAndWait 這個函數(shù),提供安全的多并發(fā)調(diào)用方式。該函數(shù)等待所有函數(shù)都結(jié)束后才返回。

// SafeGoAndWait 進行并發(fā)安全并行調(diào)用
// 第一個參數(shù)是context接口,如果還實現(xiàn)了Container接口,且綁定了日志服務(wù),則使用日志服務(wù)
// 第二個參數(shù)是匿名函數(shù)handlers數(shù)組, 進行最終的業(yè)務(wù)邏輯
// 返回handlers中任何一個錯誤(如果handlers中有業(yè)務(wù)邏輯返回錯誤)
func SafeGoAndWait(ctx context.Context, handlers ...func() error) error

調(diào)用方式參照如下的單元測試用例:

func TestSafeGoAndWait(t *testing.T) {
    container := tests.InitBaseContainer()
    container.Bind(&log.HadeTestingLogProvider{})

    errStr := "safe go test error"
    t.Log("safe go and wait start", time.Now().String())
    ctx, _ := gin.CreateTestContext(httptest.NewRecorder())

    err := goroutine.SafeGoAndWait(ctx, func() error {
        time.Sleep(1 * time.Second)
        return errors.New(errStr)
    }, func() error {
        time.Sleep(2 * time.Second)
        return nil
    }, func() error {
        time.Sleep(3 * time.Second)
        return nil
    })
    t.Log("safe go and wait end", time.Now().String())

    if err == nil {
        t.Error("err not be nil")
    } else if err.Error() != errStr {
        t.Error("err content not same")
    }

    // panic error
    err = goroutine.SafeGoAndWait(ctx, func() error {
        time.Sleep(1 * time.Second)
        return errors.New(errStr)
    }, func() error {
        time.Sleep(2 * time.Second)
        panic("test2")
    }, func() error {
        time.Sleep(3 * time.Second)
        return nil
    })
    if err == nil {
        t.Error("err not be nil")
    } else if err.Error() != errStr {
        t.Error("err content not same")
    }
}

實現(xiàn)說明

實現(xiàn)方面,有幾個難點記錄下。

首先是接口設(shè)計方面

可以看到handler函數(shù)在兩個接口中是不一樣的。在SafeGo接口中,handler定義為func() 而在SafeGoAndWait中,定義為func() error

兩者的區(qū)別就在于SafeGo這個接口是沒有能力處理error的,因為它go出去一個goroutine就直接進行接下來的操作了。而SafeGoAndWait是必須等到所有的請求結(jié)束,所以它是有能力接收到error的。

所以SafeGo的handler沒有必要設(shè)置error返回值,而SafeGoAndWait是可以設(shè)置error的。

其次是日志兼容hade

如果出現(xiàn)了panic,如何將panic的日志打印出來。

整個框架我們并不希望有任何的全局變量,包括全局的Log,所以我這里做了一個兼容邏輯。

如果只是傳遞一個context,我們就使用官方的log包進行打印。

如果傳遞的是一個即實現(xiàn)了context,又實現(xiàn)了container接口的結(jié)構(gòu),我們就從container中獲取日志服務(wù),來進行日志打印。這樣框架的所有日志就能統(tǒng)一在日志打印里面。

				if logger != nil {
						logger.Error(ctx, "safe go handler panic", map[string]interface{}{
							"stack": string(buf),
							"err":   e,
						})
				} else {
						log.Printf("panic\t%v\t%s", e, buf)
				}

由于我們修改了gin的context,讓它支持了我們的container容器結(jié)構(gòu),所以我們可以直接將gin.Context傳遞進來。具體使用起來就像這樣了:

// DemoGoroutine goroutine 的使用示例
func (api *DemoApi) DemoGoroutine(c *gin.Context) {
    logger := c.MustMakeLog()
    logger.Info(c, "request start", nil)

    // 初始化一個orm.DB
    gormService := c.MustMake(contract.ORMKey).(contract.ORMService)
    db, err := gormService.GetDB(orm.WithConfigPath("database.default"))
    if err != nil {
        logger.Error(c, err.Error(), nil)
        c.AbortWithError(50001, err)
        return
    }
    db.WithContext(c)

    err = goroutine.SafeGoAndWait(c, func() error {
        // 查詢一條數(shù)據(jù)
        queryUser := &User{ID: 1}

        err = db.First(queryUser).Error
        logger.Info(c, "query user1", map[string]interface{}{
            "err":  err,
            "name": queryUser.Name,
        })
        return err
    }, func() error {
        // 查詢一條數(shù)據(jù)
        queryUser := &User{ID: 2}

        err = db.First(queryUser).Error
        logger.Info(c, "query user2", map[string]interface{}{
            "err":  err,
            "name": queryUser.Name,
        })
        return err
    })

    if err != nil {
        c.AbortWithError(50001, err)
        return
    }
    c.JSON(200, "ok")
}

最后是打印panic的trace記錄

官方的panic其實打印的是所有g(shù)oroutine的堆棧信息。但是這里我們希望打印的是出panic的那個堆棧信息。所以我們會使用

debug.Stack()

來打印出問題的goroutine的堆棧信息。

為了打印美觀,這里將換行符統(tǒng)一替換為\n 來進行展示。

具體的實現(xiàn)代碼可以參考github地址:https://github.com/gohade/hade/blob/main/framework/util/goroutine/goroutine.go

說明文檔:https://github.com/gohade/hade/blob/main/docs/guide/util.md

總結(jié)

為hade封裝了兩個SafeGo方法。特別是第二個SafeGoAndWait,在實際工作中確實是非常有用的。

到此這篇關(guān)于如何封裝安全的go的文章就介紹到這了,更多相關(guān)封裝安全的go內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 詳解在Go語言單元測試中如何解決文件依賴問題

    詳解在Go語言單元測試中如何解決文件依賴問題

    現(xiàn)如今的?Web?應(yīng)用程序往往采用?RESTful?API?接口形式對外提供服務(wù),后端接口直接向前端返回?HTML?文件的情況越來越少,所以在程序中操作文件的場景也變少了,在編寫單元測試時,文件就成了被測試代碼的外部依賴,本文就來講解下測試過程中如何解決文件外部依賴問題
    2023-08-08
  • Go語言類型內(nèi)嵌和結(jié)構(gòu)體內(nèi)嵌的具體使用

    Go語言類型內(nèi)嵌和結(jié)構(gòu)體內(nèi)嵌的具體使用

    本文主要介紹了Go語言類型內(nèi)嵌和結(jié)構(gòu)體內(nèi)嵌的具體使用,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • 詳解如何使用Golang擴展Envoy

    詳解如何使用Golang擴展Envoy

    這篇文章主要為大家介紹了詳解如何使用Golang擴展Envoy實現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-06-06
  • Go語言中rune方法使用詳解

    Go語言中rune方法使用詳解

    本文主要介紹了Go語言中rune方法使用詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • golang代碼中調(diào)用Linux命令

    golang代碼中調(diào)用Linux命令

    本文主要介紹了golang代碼中調(diào)用Linux命令,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • 淺談go中cgo的幾種使用方式

    淺談go中cgo的幾種使用方式

    本文主要介紹了淺談go中cgo的幾種使用方式,文中根據(jù)實例編碼詳細介紹的十分詳盡,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • go語言對文件按照指定塊大小進行分割的方法

    go語言對文件按照指定塊大小進行分割的方法

    這篇文章主要介紹了go語言對文件按照指定塊大小進行分割的方法,實例分析了Go語言文件操作的技巧,需要的朋友可以參考下
    2015-03-03
  • Go讀取yaml文件到struct類的實現(xiàn)方法

    Go讀取yaml文件到struct類的實現(xiàn)方法

    本文主要介紹了Go讀取yaml文件到struct類,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-01-01
  • golang validator參數(shù)校驗的實現(xiàn)

    golang validator參數(shù)校驗的實現(xiàn)

    這篇文章主要介紹了golang validator參數(shù)校驗的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • 詳解golang 定時任務(wù)time.Sleep和time.Tick實現(xiàn)結(jié)果比較

    詳解golang 定時任務(wù)time.Sleep和time.Tick實現(xiàn)結(jié)果比較

    本文主要介紹了golang 定時任務(wù)time.Sleep和time.Tick實現(xiàn)結(jié)果比較,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-02-02

最新評論