亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go語言如何在Web服務(wù)中實現(xiàn)優(yōu)雅關(guān)機

 更新時間:2024年11月15日 10:34:34   作者:左詩右碼  
在這篇文章中,我們將通過一個簡單的例子來演示如何在 Go 語言中使用 Gin 框架實現(xiàn)優(yōu)雅關(guān)機,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下

在構(gòu)建 Web 服務(wù)時,我們往往會遇到一個棘手的問題:當(dāng)我們想要停止服務(wù)時,如何確保正在處理的請求能夠順利完成,而不是突然中斷? 這種技術(shù)被稱為“優(yōu)雅關(guān)機”,它可以確保在服務(wù)關(guān)閉時,所有的請求都被妥善處理。

在這篇文章中,我們將通過一個簡單的例子來演示如何在 Go 語言中使用 Gin 框架實現(xiàn)優(yōu)雅關(guān)機。

什么是優(yōu)雅關(guān)機?

優(yōu)雅的關(guān)機是指在關(guān)閉服務(wù)之前,先讓服務(wù)處理完當(dāng)前正在處理的請求,然后再關(guān)閉服務(wù)。這樣可以保證服務(wù)不會丟失請求,也不會影響到正在處理的請求。這種方式可以提高用戶體驗,防止服務(wù)中斷造成的數(shù)據(jù)丟失或不一致。 而執(zhí)行 Ctrl + C 或者 kill -2 pid 命令關(guān)閉服務(wù),是不會等待服務(wù)處理完請求的,這樣就會導(dǎo)致服務(wù)丟失請求。

如何實現(xiàn)優(yōu)雅的關(guān)機?

Go 1.8 版本之后,http.Server 內(nèi)置的 Shutdown() 方法就支持優(yōu)雅地關(guān)機。

代碼實現(xiàn)

我們來看一個具體的代碼示例,通過這個例子我們將展示如何實現(xiàn)優(yōu)雅關(guān)機。

package main

import (
	"context"
	"log"
	"net/http"
	"os"
	"os/signal"
	"syscall"
	"time"

	"github.com/gin-gonic/gin"
)

func main() {
	router := gin.Default()

	// 定義一個簡單的路由
	router.GET("/ping", func(c *gin.Context) {
		// 模擬一個耗時操作,比如數(shù)據(jù)庫查詢或外部API調(diào)用
		time.Sleep(5 * time.Second)
		c.JSON(http.StatusOK, gin.H{
			"message": "pong",
		})
	})

	// 配置 HTTP 服務(wù)器
	srv := &http.Server{
		Addr:    ":8080",
		Handler: router,
	}

	// 啟動服務(wù)器
	go func() {
		if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
			log.Fatalf("Failed to start server: %v", err)
		}
	}()

	// 等待中斷信號來優(yōu)雅地關(guān)閉服務(wù)器,為關(guān)閉服務(wù)器操作設(shè)置一個 5 秒的超時
	quit := make(chan os.Signal, 1) // 創(chuàng)建一個接收信號的通道

	// kill 默認(rèn)會發(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 信號轉(zhuǎn)發(fā)給 quit
	signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) // 此處不會阻塞

	<-quit                                               // 阻塞在此,當(dāng)接收到上述兩種信號時才會往下執(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)雅關(guān)閉服務(wù)(將未處理完的請求處理完再關(guān)閉服務(wù)),超過 5 秒就超時退出
	if err := srv.Shutdown(ctx); err != nil {
		log.Fatal("Server Shutdown: ", err)
	}

	log.Println("Server exiting")

}

代碼解析

1. 路由定義和服務(wù)啟動

router := gin.Default()
router.GET("/ping", func(c *gin.Context) {
	time.Sleep(5 * time.Second)
	c.JSON(http.StatusOK, gin.H{
		"message": "pong",
	})
})

首先,我們創(chuàng)建了一個簡單的 Gin 路由,并定義了一個 /ping 接口。當(dāng)訪問這個接口時,服務(wù)器會模擬一個耗時 5 秒的操作,然后返回一個 JSON 響應(yīng)。這段代碼展示了一個可能需要優(yōu)雅關(guān)機的典型場景:服務(wù)器可能正在處理耗時的請求,如果此時直接關(guān)機,請求會被中斷。

2. HTTP 服務(wù)器配置和啟動

srv := &http.Server{
	Addr:    ":8080",
	Handler: router,
}

go func() {
	if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
		log.Fatalf("Failed to start server: %v", err)
	}
}()

我們使用 http.Server 結(jié)構(gòu)體配置并啟動了一個 HTTP 服務(wù)器。服務(wù)器在一個單獨的 goroutine 中運行,這樣主程序可以繼續(xù)執(zhí)行,而不必等待服務(wù)器啟動完成。

3. 捕獲系統(tǒng)信號

quit := make(chan os.Signal, 1)
signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)

<-quit

為了實現(xiàn)優(yōu)雅關(guān)機,我們需要捕獲系統(tǒng)信號。這里使用了 os/signal 包來監(jiān)聽 syscall.SIGINTsyscall.SIGTERM 信號。當(dāng)用戶按下 Ctrl+C 或者通過 kill 命令發(fā)送信號時,這些信號會被捕獲并發(fā)送到 quit 通道,程序會隨即從阻塞狀態(tài)中恢復(fù),繼續(xù)執(zhí)行后續(xù)代碼。

4. 實現(xiàn)優(yōu)雅關(guān)機

ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel()

if err := srv.Shutdown(ctx); err != nil {
	log.Fatal("Server Shutdown: ", err)
}

在捕獲到關(guān)機信號后,我們使用 http.ServerShutdown 方法來實現(xiàn)優(yōu)雅關(guān)機。Shutdown 方法接受一個 context 參數(shù),這個 context 設(shè)置了一個超時時間。在這里,我們設(shè)置了一個 5 秒的超時時間,意味著服務(wù)器將在 5 秒內(nèi)等待未完成的請求處理完畢,然后關(guān)閉。如果超過了設(shè)定的超時時間,服務(wù)器將退出,程序也會正常結(jié)束。

如何驗證優(yōu)雅關(guān)機的效果?

要驗證優(yōu)雅關(guān)機的效果,可以按照以下步驟操作:

  • 打開終端,運行 go run gin_shutdown.go
  • 打開瀏覽器,并訪問 http://127.0.0.1:8080/ping 此時瀏覽器應(yīng)該會白屏等待服務(wù)端返回響應(yīng)
  • 在剛剛打開的終端上迅速按下 Ctrl+C 命令,此時會自動給程序發(fā)送 syscall.SIGINT 信號
  • 此時程序并不會立即退出,而是會等上面的第 2 步的響應(yīng)返回之后再退出,從而實現(xiàn)優(yōu)雅關(guān)機的效果

總結(jié)

優(yōu)雅關(guān)機是構(gòu)建健壯 Web 服務(wù)的一個重要技術(shù)點,它確保了在服務(wù)關(guān)閉時所有正在處理的請求都能被妥善完成。在本文中,我們通過 Gin 框架演示了如何在 Go 中實現(xiàn)優(yōu)雅關(guān)機。通過這種方式,我們可以提升用戶體驗,減少由于服務(wù)中斷導(dǎo)致的各種潛在問題。

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

相關(guān)文章

  • Go 如何批量修改文件名

    Go 如何批量修改文件名

    這篇文章主要介紹了Go 批量修改文件名的實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • Go語言實現(xiàn)IP段范圍校驗示例

    Go語言實現(xiàn)IP段范圍校驗示例

    這篇文章主要介紹了Go語言實現(xiàn)IP段范圍校驗示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-09-09
  • 解決Go語言中高頻次和高并發(fā)下隨機數(shù)重復(fù)的問題

    解決Go語言中高頻次和高并發(fā)下隨機數(shù)重復(fù)的問題

    在Golang中,獲取隨機數(shù)的方法一般會介紹有兩種,一種是基于math/rand的偽隨機,一種是基于crypto/rand的真隨機,math/rand由于其偽隨機的原理,經(jīng)常會出現(xiàn)重復(fù)的隨機數(shù),導(dǎo)致在需要進行隨機的業(yè)務(wù)出現(xiàn)較多的重復(fù)問題,所以本文給大家介紹了較好的解放方案
    2023-12-12
  • 圖解Golang的GC垃圾回收算法

    圖解Golang的GC垃圾回收算法

    這篇文章主要介紹了圖解Golang的GC垃圾回收算法,詳細(xì)的介紹了三種經(jīng)典的算法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-03-03
  • 使用gorm.Scopes函數(shù)實現(xiàn)復(fù)用查詢邏輯示例

    使用gorm.Scopes函數(shù)實現(xiàn)復(fù)用查詢邏輯示例

    這篇文章主要為大家介紹了使用gorm.Scopes函數(shù)實現(xiàn)復(fù)用查詢邏輯示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-12-12
  • uber go zap 日志框架支持異步日志輸出

    uber go zap 日志框架支持異步日志輸出

    這篇文章主要為大家介紹了uber go zap 日志框架支持異步日志輸出示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-01-01
  • GoLang調(diào)用鏈可視化go-callvis使用介紹

    GoLang調(diào)用鏈可視化go-callvis使用介紹

    與鏈路追蹤(Tracing)不同,Tracing關(guān)注復(fù)雜的分布式環(huán)境中各個服務(wù)節(jié)點間的調(diào)用關(guān)系,主要用于服務(wù)治理。而我們本次探索的代碼調(diào)用鏈路則是代碼方法級別的調(diào)用關(guān)系,主要用于代碼設(shè)計
    2023-02-02
  • golang 40行代碼實現(xiàn)通用協(xié)程池

    golang 40行代碼實現(xiàn)通用協(xié)程池

    golang協(xié)程機制很方便的解決了并發(fā)編程的問題,但是協(xié)程并不是沒有開銷的,所以也需要適當(dāng)限制一下數(shù)量。這篇文章主要介紹了golang 40行代碼實現(xiàn)通用協(xié)程池,需要的朋友可以參考下
    2018-08-08
  • golang mapstructure庫的具體使用

    golang mapstructure庫的具體使用

    mapstructure用于將通用的map[string]interface{}解碼到對應(yīng)的 Go 結(jié)構(gòu)體中,或者執(zhí)行相反的操作,本文主要介紹了golang mapstructure庫的具體使用,感興趣的可以了解一下
    2023-09-09
  • logrus日志自定義格式操作

    logrus日志自定義格式操作

    這篇文章主要介紹了logrus日志自定義格式操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-11-11

最新評論