使用golang如何優(yōu)雅的關機或重啟操作示例
前言
我們編寫的Web項目部署之后,經(jīng)常會因為需要進行配置變更或功能迭代而重啟服務,單純的kill -9 pid的方式會強制關閉進程,這樣就會導致服務端當前正在處理的請求失敗,那有沒有更優(yōu)雅的方式來實現(xiàn)關機或重啟呢?
閱讀本文需要了解一些UNIX系統(tǒng)中信號的概念,請?zhí)崆安殚嗁Y料預習。
優(yōu)雅地關機
什么是優(yōu)雅關機?
優(yōu)雅關機就是服務端關機命令發(fā)出后不是立即關機,而是等待當前還在處理的請求全部處理完畢后再退出程序,是一種對客戶端友好的關機方式。而執(zhí)行Ctrl+C關閉服務端時,會強制結束進程導致正在訪問的請求出現(xiàn)問題。
如何實現(xiàn)優(yōu)雅關機?
Go 1.8版本之后, http.Server 內(nèi)置的 Shutdown() 方法就支持優(yōu)雅地關機,具體示例如下:
// +build go1.8
package main
import (
"context"
"log"
"net/http"
"os"
"os/signal"
"syscall"
"time"
"github.com/gin-gonic/gin"
)
func main() {
router := gin.Default()
router.GET("/", func(c *gin.Context) {
time.Sleep(5 * time.Second)
c.String(http.StatusOK, "Welcome Gin Server")
})
srv := &http.Server{
Addr: ":8080",
Handler: router,
}
go func() {
// 開啟一個goroutine啟動服務
if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
log.Fatalf("listen: %s\n", err)
}
}()
// 等待中斷信號來優(yōu)雅地關閉服務器,為關閉服務器操作設置一個5秒的超時
quit := make(chan os.Signal, 1) // 創(chuàng)建一個接收信號的通道
// kill 默認會發(fā)送 syscall.SIGTERM 信號
// kill -2 發(fā)送 syscall.SIGINT 信號,我們常用的Ctrl+C就是觸發(fā)系統(tǒng)SIGINT信號
// kill -9 發(fā)送 syscall.SIGKILL 信號,但是不能被捕獲,所以不需要添加它
// signal.Notify把收到的 syscall.SIGINT或syscall.SIGTERM 信號轉發(fā)給quit
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) // 此處不會阻塞
<-quit // 阻塞在此,當接收到上述兩種信號時才會往下執(zhí)行
log.Println("Shutdown Server ...")
// 創(chuàng)建一個5秒超時的context
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()
// 5秒內(nèi)優(yōu)雅關閉服務(將未處理完的請求處理完再關閉服務),超過5秒就超時退出
if err := srv.Shutdown(ctx); err != nil {
log.Fatal("Server Shutdown: ", err)
}
log.Println("Server exiting")
}
如何驗證優(yōu)雅關機的效果呢?
上面的代碼運行后會在本地的8080端口開啟一個web服務,它只注冊了一條路由/,后端服務會先sleep 5秒鐘然后才返回響應信息。
我們按下Ctrl+C時會發(fā)送syscall.SIGINT來通知程序優(yōu)雅關機,具體做法如下:
- 打開終端,編譯并執(zhí)行上面的代碼
- 打開一個瀏覽器,訪問127.0.0.1:8080/,此時瀏覽器白屏等待服務端返回響應。
- 在終端迅速執(zhí)行
Ctrl+C命令給程序發(fā)送syscall.SIGINT信號 - 此時程序并不立即退出而是等我們第2步的響應返回之后再退出,從而實現(xiàn)優(yōu)雅關機。
優(yōu)雅地重啟
優(yōu)雅關機實現(xiàn)了,那么該如何實現(xiàn)優(yōu)雅重啟呢?
我們可以使用 fvbock/endless 來替換默認的 ListenAndServe啟動服務來實現(xiàn), 示例代碼如下:
package main
import (
"log"
"net/http"
"time"
"github.com/fvbock/endless"
"github.com/gin-gonic/gin"
)
func main() {
router := gin.Default()
router.GET("/", func(c *gin.Context) {
time.Sleep(5 * time.Second)
c.String(http.StatusOK, "hello gin!")
})
// 默認endless服務器會監(jiān)聽下列信號:
// syscall.SIGHUP,syscall.SIGUSR1,syscall.SIGUSR2,syscall.SIGINT,syscall.SIGTERM和syscall.SIGTSTP
// 接收到 SIGHUP 信號將觸發(fā)`fork/restart` 實現(xiàn)優(yōu)雅重啟(kill -1 pid會發(fā)送SIGHUP信號)
// 接收到 syscall.SIGINT或syscall.SIGTERM 信號將觸發(fā)優(yōu)雅關機
// 接收到 SIGUSR2 信號將觸發(fā)HammerTime
// SIGUSR1 和 SIGTSTP 被用來觸發(fā)一些用戶自定義的hook函數(shù)
if err := endless.ListenAndServe(":8080", router); err!=nil{
log.Fatalf("listen: %s\n", err)
}
log.Println("Server exiting")
}
如何驗證優(yōu)雅重啟的效果呢?
我們通過執(zhí)行kill -1 pid命令發(fā)送syscall.SIGINT來通知程序優(yōu)雅重啟,具體做法如下:
- 打開終端,go build -o graceful_restart編譯并執(zhí)行./graceful_restart,終端輸出當前pid(假設為43682)
- 將代碼中處理請求函數(shù)返回的
hello gin!修改為hello q1mi!,再次編譯go build -o graceful_restart - 打開一個瀏覽器,訪問127.0.0.1:8080/,此時瀏覽器白屏等待服務端返回響應。
- 在終端迅速執(zhí)行kill -1 43682命令給程序發(fā)送syscall.SIGHUP信號
- 等第3步瀏覽器收到響應信息
hello gin!后再次訪問127.0.0.1:8080/會收到hello q1mi!的響應。 - 在不影響當前未處理完請求的同時完成了程序代碼的替換,實現(xiàn)了優(yōu)雅重啟。
但是需要注意的是,此時程序的PID變化了,因為endless 是通過fork子進程處理新請求,待原進程處理完當前請求后再退出的方式實現(xiàn)優(yōu)雅重啟的。所以當你的項目是使用類似supervisor的軟件管理進程時就不適用這種方式了。
總結
無論是優(yōu)雅關機還是優(yōu)雅重啟歸根結底都是通過監(jiān)聽特定系統(tǒng)信號,然后執(zhí)行一定的邏輯處理保障當前系統(tǒng)正在處理的請求被正常處理后再關閉當前進程。使用優(yōu)雅關機還是使用優(yōu)雅重啟以及怎么實現(xiàn),這就需要根據(jù)項目實際情況來決定了。
以上就是使用golang如何優(yōu)雅的關機或重啟 的詳細內(nèi)容,更多關于golang關機重啟 的資料請關注腳本之家其它相關文章!
相關文章
Golang Printf,Sprintf,Fprintf 格式化詳解
這篇文章主要介紹了Golang Printf,Sprintf,Fprintf 格式化詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-03-03
Go調(diào)用opencv實現(xiàn)圖片矯正的代碼示例
這篇文章主要為大家詳細介紹了Go調(diào)用opencv實現(xiàn)圖片矯正的代碼示例,文中的示例代碼簡潔易懂,感興趣的小伙伴可以跟隨小編一起學習一下2023-09-09
Golang使用http協(xié)議實現(xiàn)心跳檢測程序過程詳解
這篇文章主要介紹了Golang使用http協(xié)議實現(xiàn)心跳檢測程序過程,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習吧2023-03-03
Go語言中make和new函數(shù)的用法與區(qū)別
這篇文章介紹了Go語言中make和new函數(shù)的用法與區(qū)別,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-07-07
golang 調(diào)用c語言動態(tài)庫方式實現(xiàn)
本文主要介紹了golang 調(diào)用c語言動態(tài)庫方式實現(xiàn),文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-12-12
go語言開發(fā)中如何優(yōu)雅得關閉協(xié)程方法
這篇文章主要為大家介紹了go語言開發(fā)中如何優(yōu)雅得關閉協(xié)程方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-05-05

