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

go實現(xiàn)服務(wù)優(yōu)雅關(guān)閉的示例

 更新時間:2023年02月09日 09:44:28   作者:亞洲第一中鋒_哈達迪  
本文主要介紹了go實現(xiàn)服務(wù)優(yōu)雅關(guān)閉的示例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

為什么需要優(yōu)雅關(guān)閉

什么叫優(yōu)雅關(guān)閉?先說不優(yōu)雅關(guān)閉,就是什么都不管,強制關(guān)閉進程,這會導(dǎo)致有些正在處理中的請求被強行中斷

這樣做有什么問題?

  • 用戶本次請求會失敗,降低用戶體驗
  • 沒有事務(wù)的數(shù)據(jù)庫操作,會產(chǎn)生部分成功的問題,破壞原子性
  • 某些緩服務(wù)需要定期將本地緩存刷到遠程db,強制關(guān)閉會導(dǎo)致數(shù)據(jù)丟失

優(yōu)雅關(guān)閉的核心是以下功能:

  • 如何監(jiān)聽退出信號
  • 如何拒絕新請求
  • 如何等待進行中的請求處理完畢

監(jiān)控服務(wù)退出信號

在go中使用下面的代碼監(jiān)聽退出信號,如果c返回,說明監(jiān)聽到信號

不同的操作系統(tǒng)監(jiān)聽不同的退出信號

c := make(chan os.Signal, 1)
signals := []os.Signal{
   syscall.SIGINT,
   syscall.SIGTERM,
   syscall.SIGQUIT,
}

signal.Notify(c, signals...)
<-c

拒絕新請求

go在1.8后增加了shutdown方法來,我們看看它怎么實現(xiàn)優(yōu)雅關(guān)閉:

srv.inShutdown.setTrue()
lnerr := srv.closeListenersLocked()
srv.closeDoneChanLocked()
  • 設(shè)置inShutdown標志位
  • 關(guān)閉所有的listener
  • 關(guān)閉doneChan

這一段對應(yīng)到http服務(wù)接收請求的流程:

for {
   rw, err := l.Accept()
   if err != nil {
      select {
      case <-srv.getDoneChan():
         return ErrServerClosed
   // ...       
}

一旦關(guān)閉listener,關(guān)閉doneChan后,http服務(wù)就不會再接收新的請求,直接返回

執(zhí)行關(guān)閉之前的回調(diào)

for _, f := range srv.onShutdown {
   go f()
}

這里的回調(diào)實現(xiàn)得比較粗糙:

  • 沒有優(yōu)先級的概念,所有回調(diào)并發(fā)執(zhí)行,因此需要保證回調(diào)之間沒有依賴
  • 雖然回調(diào)不適合長時間運行,但Go http沒有提供機制來保證這些回調(diào)一定能執(zhí)行完畢,若想做到這點需要自己處理

等待處理中的請求執(zhí)行完畢

設(shè)置標識位可以拒絕新的請求,但依舊在執(zhí)行的請求還在處理中,需要等這些請求執(zhí)行完畢

等待處理中的請求執(zhí)行完畢有兩種思路:

  • 等待一段固定的時間
  • 實時維護請求的計數(shù)

go選擇了兩種方式結(jié)合的模式,通過ctx設(shè)置一個最大的等待時間,同時不斷輪詢正在請求中的計數(shù)

ctx超時或者計數(shù)變?yōu)?,都會返回

timer := time.NewTimer(nextPollInterval())
defer timer.Stop()
for {
   if srv.closeIdleConns() && srv.numListeners() == 0 {
      return lnerr
   }
   select {
   case <-ctx.Done():
      return ctx.Err()
   case <-timer.C:
      timer.Reset(nextPollInterval())
   }
}

這里每隔一定時間檢查已有請求是否執(zhí)行完畢,如果執(zhí)行完畢,或者外部通過ctx設(shè)置的超時到期就會返回

檢查間隔是多少?

  • 從1ms開始,每輪檢查后倍增,最大500ms

怎么判斷是否執(zhí)行完畢?

  • 所有的連接都關(guān)閉
  • 所有的listener都關(guān)閉

服務(wù)收到監(jiān)聽信號返回之前,關(guān)閉連接和listener,會被這里檢查到

在這里插入圖片描述

實戰(zhàn)

func main() {
   // 注冊路由
   http.Handle("/aaa", http.HandlerFunc(func(writer http.ResponseWriter, req *http.Request) {
      time.Sleep(time.Second * 10)
      fmt.Println(111)
   }))
   server := http.Server{
      Addr:    "localhost:8080",
      Handler: http.DefaultServeMux,
   }

   close := make(chan int)
   go func() {
      quit := make(chan os.Signal)
      signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT)
      <-quit

      ctx, cancel := context.WithTimeout(context.Background(), time.Second*30)
      defer cancel()
      err := server.Shutdown(ctx)
      log.Print(err)
      
      // 控制外層退出
      close <- 1
   }()

   err := server.ListenAndServe()
   fmt.Println(err)

   <-close
}

該代碼做了下面的事:

  • 注冊一個10s才返回的路由處理函數(shù)
  • 開子協(xié)程監(jiān)聽OS的退出信號,如果監(jiān)聽到了開始進行優(yōu)雅關(guān)閉,雖多等待30s
  • 主協(xié)程調(diào)用 server.ListenAndServe(),開始監(jiān)聽請求

需要注意的是,一定要在子協(xié)程中優(yōu)雅關(guān)閉結(jié)束后,主協(xié)程才能退出,這里用channel控制

因為主協(xié)程發(fā)現(xiàn)doneChan被關(guān)閉時會馬上返回,但此時主協(xié)程開的業(yè)務(wù)處理協(xié)程還在進行中,如果主協(xié)程此時退出,無法達到優(yōu)雅關(guān)閉的效果

按照以下流程測試:

  • 啟動 Web 服務(wù)
  • 在瀏覽器請求http://localhost:8080/aaa
  • 過5秒后在控制臺按下ctrl+c
  • 觀察控制臺程序是否不會立刻結(jié)束,而是在 10s 后結(jié)束

支持強制退出

既然有優(yōu)雅退出,那就有強制退出,我們假設(shè)如果按下兩次ctrl+c,代表用戶希望服務(wù)強制退出:

close := make(chan int, 2)
go func() {
   quit := make(chan os.Signal)
   signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT)
   <-quit
   go func() {
      <-quit
      os.Exit(1)
   }()

   // ...
}()

做法很簡單,收到第一個退出信號后,再開一個子協(xié)程,如果再收到退出信號,就調(diào)用os.Exit退出進程

并且close channel的容量需要為2,避免當兩次退出信號過短時丟失信號

到此這篇關(guān)于go實現(xiàn)服務(wù)優(yōu)雅關(guān)閉的示例的文章就介紹到這了,更多相關(guān)go 服務(wù)關(guān)閉內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 淺析GO語言的垃圾回收機制

    淺析GO語言的垃圾回收機制

    今天我們來聊聊golang是如何進行垃圾回收的,我們知道,目前各語言進行垃圾回收的方法有很多,如引用計數(shù)、標記清除、分代回收、三色標記等,各種方式都有其特點,文中介紹的非常詳細,感興趣的小伙伴跟著小編一起學(xué)習(xí)吧
    2023-07-07
  • golang?gorm的預(yù)加載及軟刪硬刪的數(shù)據(jù)操作示例

    golang?gorm的預(yù)加載及軟刪硬刪的數(shù)據(jù)操作示例

    這篇文章主要介紹了golang?gorm的預(yù)加載及軟刪硬刪的數(shù)據(jù)操作示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪
    2022-04-04
  • Go錯誤和異常CGO?fallthrough處理教程詳解

    Go錯誤和異常CGO?fallthrough處理教程詳解

    這篇文章主要為大家介紹了Go錯誤和異常CGO?fallthrough使用教程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • Golang實現(xiàn)單元測試中的接口層

    Golang實現(xiàn)單元測試中的接口層

    接口層主要負責(zé)的就是請求的處理,最常見的就是?HTTP?請求的處理。這篇文章主要為大家介紹了Golang如何實現(xiàn)單元測試中的接口層,需要的可以參考一下
    2023-03-03
  • Golang使用Swagger文檔教程詳解

    Golang使用Swagger文檔教程詳解

    這篇文章主要為大家詳細介紹Golang使用Swagger文檔教程的相關(guān)知識,文中通過示例代碼講解詳細,具有一定的借鑒價值,感興趣的小伙伴可以學(xué)習(xí)一下
    2023-12-12
  • 攔截信號Golang應(yīng)用優(yōu)雅關(guān)閉的操作方法

    攔截信號Golang應(yīng)用優(yōu)雅關(guān)閉的操作方法

    這篇文章主要介紹了攔截信號優(yōu)雅關(guān)閉Golang應(yīng)用,本文介紹了信號的概念及常用信號,并給出了應(yīng)用廣泛的幾個示例,例如優(yōu)雅地關(guān)閉應(yīng)用服務(wù)、在命令行應(yīng)用中接收終止命令,需要的朋友可以參考下
    2023-02-02
  • 盤點幾種Go語言開發(fā)的IDE

    盤點幾種Go語言開發(fā)的IDE

    Go語言作為一種新興的編程語言,近年來受到了越來越多的關(guān)注,它以其簡潔、高效和并發(fā)性能而聞名,被廣泛應(yīng)用于各種軟件開發(fā)項目中,本文將介紹幾種常用的Go語言IDE,并對它們進行比較,幫助開發(fā)者根據(jù)自己的需求選擇合適的工具,需要的朋友可以參考下
    2023-11-11
  • Golang使用channel實現(xiàn)一個優(yōu)雅退出功能

    Golang使用channel實現(xiàn)一個優(yōu)雅退出功能

    最近補?Golang?channel?方面八股的時候發(fā)現(xiàn)用?channel?實現(xiàn)一個優(yōu)雅退出功能好像不是很難,之前寫的?HTTP?框架剛好也不支持優(yōu)雅退出功能,于是就參考了?Hertz?優(yōu)雅退出方面的代碼,為我的?PIANO?補足了這個?feature
    2023-03-03
  • Go庫text與template包使用示例詳解

    Go庫text與template包使用示例詳解

    這篇文章主要為大家介紹了Go庫text與template包使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-12-12
  • Golang 實現(xiàn)獲取當前函數(shù)名稱和文件行號等操作

    Golang 實現(xiàn)獲取當前函數(shù)名稱和文件行號等操作

    這篇文章主要介紹了Golang 實現(xiàn)獲取當前函數(shù)名稱和文件行號等操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05

最新評論