婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁(yè) > 知識(shí)庫(kù) > golang的httpserver優(yōu)雅重啟方法詳解

golang的httpserver優(yōu)雅重啟方法詳解

熱門(mén)標(biāo)簽:黃岡人工智能電銷(xiāo)機(jī)器人哪個(gè)好 濱州自動(dòng)電銷(xiāo)機(jī)器人排名 汕頭小型外呼系統(tǒng) 建造者2地圖標(biāo)注 阿里云ai電話(huà)機(jī)器人 浙江高頻外呼系統(tǒng)多少錢(qián)一個(gè)月 鄭州亮點(diǎn)科技用的什么外呼系統(tǒng) 釘釘有地圖標(biāo)注功能嗎 惠州電銷(xiāo)防封電話(huà)卡

前言

去年在做golangserver的時(shí)候,內(nèi)部比較頭疼的就是在線(xiàn)服務(wù)發(fā)布的時(shí)候,大量用戶(hù)的請(qǐng)求在發(fā)布時(shí)候會(huì)被重連,在那時(shí)候也想了n多的方法,最后還是落在一個(gè)github上的項(xiàng)目,facebook的一個(gè)golang項(xiàng)目grace,那時(shí)候簡(jiǎn)單研究測(cè)試了一下可以就直接在內(nèi)部使用了起來(lái),這段時(shí)間突然想起來(lái),又想仔細(xì)研究一下這個(gè)項(xiàng)目了。

從原理上來(lái)說(shuō)是這樣一個(gè)過(guò)程:

    1)發(fā)布新的bin文件去覆蓋老的bin文件

    2)發(fā)送一個(gè)信號(hào)量,告訴正在運(yùn)行的進(jìn)程,進(jìn)行重啟

    3)正在運(yùn)行的進(jìn)程收到信號(hào)后,會(huì)以子進(jìn)程的方式啟動(dòng)新的bin文件

    4)新進(jìn)程接受新請(qǐng)求,并處理

    5)老進(jìn)程不再接受請(qǐng)求,但是要等正在處理的請(qǐng)求處理完成,所有在處理的請(qǐng)求處理完之后,便自動(dòng)退出

    6)新進(jìn)程在老進(jìn)程退出之后,由init進(jìn)程收養(yǎng),但是會(huì)繼續(xù)服務(wù)。

所以一步一步來(lái)看,關(guān)鍵是從第2步開(kāi)始之后怎么做,所以我們先來(lái)看看第2步的實(shí)現(xiàn),這個(gè)應(yīng)該說(shuō)很簡(jiǎn)單,發(fā)送信號(hào)量到一個(gè)進(jìn)程,使用kill命令即可,在facebook這個(gè)項(xiàng)目中發(fā)送的信號(hào)量有3個(gè):SIGINT,SIGTERM,SIGUSR2,前面兩個(gè)信號(hào)收到后程序會(huì)直接退出,后面一個(gè)信號(hào)SIGUSR2才會(huì)執(zhí)行所謂的優(yōu)雅重啟。

第3步,正在運(yùn)行的進(jìn)程收到SIGUSR2信號(hào)后,會(huì)以子進(jìn)程的方式啟動(dòng)新的bin文件。先直接上代碼看:https://github.com/facebookgo/grace/blob/master/gracehttp/http.go

func (a *app) signalHandler(wg *sync.WaitGroup) {
 ch := make(chan os.Signal, 10)
 signal.Notify(ch, syscall.SIGINT, syscall.SIGTERM, syscall.SIGUSR2)
 for {
  sig := -ch
  switch sig {
  case syscall.SIGINT, syscall.SIGTERM: 
   // this ensures a subsequent INT/TERM will trigger standard go behaviour of
   // terminating. 執(zhí)行標(biāo)準(zhǔn)的go終止行為,程序就結(jié)束了
   signal.Stop(ch)
   a.term(wg)
   return
  case syscall.SIGUSR2: // 這里開(kāi)始執(zhí)行優(yōu)雅重啟
   err := a.preStartProcess() 
   // 這個(gè)函數(shù)在源代碼中沒(méi)有具體實(shí)現(xiàn)功能,只是預(yù)留了一個(gè)鉤子函數(shù),用戶(hù)可以注冊(cè)自己的函數(shù),可以在重啟之前做些自定義的事情。一般情況下也沒(méi)有什么可以做的,除非有些特殊的服務(wù)環(huán)境或是狀態(tài)保存之類(lèi)的,至少目前,我們的server還沒(méi)有遇到
   if err != nil {
    a.errors - err
   }
   // we only return here if there's an error, otherwise the new process
   // will send us a TERM when it's ready to trigger the actual shutdown.
   if _, err := a.net.StartProcess(); err != nil { // 這里開(kāi)始正式所謂的優(yōu)雅重啟   
    a.errors - err
   }
  }
 }
}

a.net.StartProcess的過(guò)程我們來(lái)看看基本過(guò)程:

func (n *Net) StartProcess() (int, error) {
 listeners, err := n.activeListeners() // 獲取目前在監(jiān)聽(tīng)的端口,這塊也是重點(diǎn),下面重點(diǎn)介紹
 if err != nil {
  return 0, err
 }
 
 // Extract the fds from the listeners. 從監(jiān)聽(tīng)端口中把文件描述符取出來(lái)
 files := make([]*os.File, len(listeners))
 for i, l := range listeners {
  files[i], err = l.(filer).File()
  if err != nil {
   return 0, err
  }
  defer files[i].Close()
 }
 
 // Use the original binary location. This works with symlinks such that if
 // the file it points to has been changed we will use the updated symlink.
 // 獲取可執(zhí)行bin文件的路勁,也可以是鏈接路勁,會(huì)使用最新的鏈接路徑作為啟動(dòng)文件路勁的
 argv0, err := exec.LookPath(os.Args[0])
 if err != nil {
  return 0, err
 }
 
 // Pass on the environment and replace the old count key with the new one.
 // 獲取 LISTEN_FDS 換進(jìn)變量值 
 var env []string
 for _, v := range os.Environ() {
  if !strings.HasPrefix(v, envCountKeyPrefix) {
   env = append(env, v)
  }
 }
 env = append(env, fmt.Sprintf("%s%d", envCountKeyPrefix, len(listeners)))
 
 allFiles := append([]*os.File{os.Stdin, os.Stdout, os.Stderr}, files...)
 // 這里調(diào)用一個(gè)golang底層的進(jìn)程啟動(dòng)函數(shù),來(lái)指定,上面獲取的參數(shù)來(lái)啟動(dòng)進(jìn)程
 process, err := os.StartProcess(argv0, os.Args, os.ProcAttr{
  Dir: originalWD,
  Env: env,
  Files: allFiles,
 })
 if err != nil {
  return 0, err
 }
 // 返回新進(jìn)程id。
 return process.Pid, nil 
}

以上是啟動(dòng)新進(jìn)程,并且接管監(jiān)聽(tīng)端口的過(guò)程, 一般情況下端口是不可以重復(fù)監(jiān)聽(tīng)的,所以這里就要需要使用比較特別的辦法,從上面的代碼來(lái)看就是讀取監(jiān)聽(tīng)端口的文件描述符,并且把監(jiān)聽(tīng)端口的文件描述符傳遞給子進(jìn)程,子進(jìn)程里從這個(gè)文件描述符實(shí)現(xiàn)對(duì)端口的監(jiān)聽(tīng)

另外還有一個(gè)比較特別的地方就是老的接口怎么關(guān)閉的問(wèn)題,關(guān)閉必須要把已經(jīng)收到的請(qǐng)求處理完成之后再關(guān)閉。為此facebook的同學(xué)另外開(kāi)了一個(gè)項(xiàng)目httpdown,繼承了原始的httpserver,但是多了對(duì)各種鏈接狀態(tài)的維護(hù)和處理,這部分后面在分析。

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

您可能感興趣的文章:
  • [Asp.Net Core]用Blazor Server Side實(shí)現(xiàn)圖片驗(yàn)證碼
  • [Asp.Net Core] 淺談Blazor Server Side
  • Ant Design Blazor 組件庫(kù)的路由復(fù)用多標(biāo)簽頁(yè)功能
  • HTTP中header頭部信息詳解
  • Golang簡(jiǎn)單實(shí)現(xiàn)http的server端和client端
  • IOS利用CocoaHttpServer搭建手機(jī)本地服務(wù)器
  • Golang實(shí)現(xiàn)http server提供壓縮文件下載功能
  • 在Golang中使用http.FileServer返回靜態(tài)文件的操作
  • 基于http.server搭建局域網(wǎng)服務(wù)器過(guò)程解析
  • Blazor Server 應(yīng)用程序中進(jìn)行 HTTP 請(qǐng)求

標(biāo)簽:瀘州 滄州 泰安 晉中 阿壩 昭通 駐馬店 東營(yíng)

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《golang的httpserver優(yōu)雅重啟方法詳解》,本文關(guān)鍵詞  golang,的,httpserver,優(yōu)雅,重啟,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《golang的httpserver優(yōu)雅重啟方法詳解》相關(guān)的同類(lèi)信息!
  • 本頁(yè)收集關(guān)于golang的httpserver優(yōu)雅重啟方法詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    主站蜘蛛池模板: 专栏| 科尔| 独山县| 庐江县| 偃师市| 安福县| 九龙坡区| 邛崃市| 大余县| 托里县| 顺昌县| 左贡县| 婺源县| 高平市| 南康市| 天祝| 翁牛特旗| 遵义县| 伊宁市| 台南县| 嘉峪关市| 乌兰县| 都安| 全州县| 上栗县| 开原市| 赣榆县| 东乡族自治县| 莲花县| 平乡县| 扶绥县| 沿河| 福安市| 册亨县| 万盛区| 灵武市| 彩票| 军事| 黔南| 五华县| 屏东县|