go實現(xiàn)服務(wù)優(yōu)雅關(guān)閉的示例
為什么需要優(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)文章
golang?gorm的預(yù)加載及軟刪硬刪的數(shù)據(jù)操作示例
這篇文章主要介紹了golang?gorm的預(yù)加載及軟刪硬刪的數(shù)據(jù)操作示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步早日升職加薪2022-04-04攔截信號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-02Golang使用channel實現(xiàn)一個優(yōu)雅退出功能
最近補?Golang?channel?方面八股的時候發(fā)現(xiàn)用?channel?實現(xiàn)一個優(yōu)雅退出功能好像不是很難,之前寫的?HTTP?框架剛好也不支持優(yōu)雅退出功能,于是就參考了?Hertz?優(yōu)雅退出方面的代碼,為我的?PIANO?補足了這個?feature2023-03-03Golang 實現(xiàn)獲取當前函數(shù)名稱和文件行號等操作
這篇文章主要介紹了Golang 實現(xiàn)獲取當前函數(shù)名稱和文件行號等操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-05-05