中文字幕理论片,69视频免费在线观看,亚洲成人app,国产1级毛片,刘涛最大尺度戏视频,欧美亚洲美女视频,2021韩国美女仙女屋vip视频

打開(kāi)APP
userphoto
未登錄

開(kāi)通VIP,暢享免費(fèi)電子書等14項(xiàng)超值服

開(kāi)通VIP
golang API開(kāi)發(fā)過(guò)程的中的自動(dòng)重啟(基于gin框架)

概要

基于 golang Gin 框架開(kāi)發(fā) web 服務(wù)時(shí), 需要時(shí)不時(shí)的 go build , 然后重啟服務(wù)查看運(yùn)行結(jié)果.

go build 的過(guò)程集成在編輯器中(emacs), 可以通過(guò)快捷鍵迅速完成, 但是每次重啟服務(wù)都切換到命令行中操作.

因此, 希望能夠編譯通過(guò)之后自動(dòng)重啟服務(wù).

這里并不是部署階段的服務(wù)重啟, 所以不用過(guò)多考慮是否正常退出其中的協(xié)程.

實(shí)現(xiàn)方式

在開(kāi)源的 illuminant 項(xiàng)目中, 已經(jīng)將相應(yīng)的代碼集成到 gin 的 debug mode 中.

代碼文件: https://gitee.com/wangyubin/illuminant/blob/dev/server_cmd.go 1  func setupWatcher() (chan struct{}, error) {  2    file, err := osext.Executable()  3    if err != nil {  4      return nil, err  5    }  6    log.Printf("watching %q\n", file)  7    w, err := fsnotify.NewWatcher()  8    if err != nil {  9      return nil, err 10    } 11    done := make(chan struct{}) 12    go func() { 13      select { 14      case e := <-w.Events: 15        log.Printf("watcher received: %+v", e) 16        err := syscall.Exec(file, os.Args, os.Environ()) 17        if err != nil { 18          log.Fatal(err) 19        } 20      case err := <-w.Errors: 21        log.Printf("watcher error: %+v", err) 22      case <-done: 23        log.Print("watcher shutting down") 24        return 25      } 26    }() 27    err = w.Add(file) 28    if err != nil { 29      return nil, err 30    } 31    return done, nil 32  }

在 gin debug mode 下, 使用此方法自動(dòng)重啟服務(wù) 1  if c.Bool("prod") {  2    gin.SetMode(gin.ReleaseMode)  3    // start route  4    return routes.Routes(cnf.Server.Port)  5  } else {  6    gin.SetMode(gin.DebugMode)  7    watcher, err := setupWatcher()  8    if err != nil {  9      // do something sensible 10      log.Fatal(err) 11    } 12    defer close(watcher) 13    return routes.Routes(cnf.Server.Port) 14  }

補(bǔ)充

上面函數(shù)的核心有以下兩點(diǎn):

w, err := fsnotify.NewWatcher(): 創(chuàng)建監(jiān)控文件變化的 watcher, err = w.Add(file) 并將當(dāng)前二進(jìn)制文件加入到監(jiān)控文件列表中

err := syscall.Exec(file, os.Args, os.Environ()) 接受到文件變化的事件時(shí), 重新調(diào)用一次自己, 使用上次一樣的參數(shù)和環(huán)境變量

syscall.Exec

對(duì)于這個(gè)函數(shù), 一般可能用的比較少, 這里稍微介紹下. 它有 3 個(gè)參數(shù):

args[0]: 可執(zhí)行文件的路徑(相對(duì)路徑, 絕對(duì)路徑或者 PATH 中的路徑都可以)

args[1]: 命令的參數(shù)

args[2]: 命令的執(zhí)行的環(huán)境變量, os.Environ() 表示繼承 caller 的環(huán)境變量

當(dāng) syscall.Exec 執(zhí)行時(shí), 在它之前的所有未執(zhí)行完的程序都會(huì)被中止(包括在 go routine 中執(zhí)行的程序),

然后執(zhí)行 syscall.Exec 調(diào)用的命令, 該命令還保持在之前程序的 PID 下執(zhí)行.

syscall.Exec 是最后一條執(zhí)行的代碼, 重啟時(shí)在它之后可以有代碼, 但是都不會(huì)被執(zhí)行到, 包括 defer 中的代碼.

下面是個(gè)小例子(通過(guò)這個(gè)例子可以驗(yàn)證上面的結(jié)論): 1  package main  2    3  import (  4   "fmt"  5   "log"  6   "os"  7   "syscall"  8   "time"  9   10   "github.com/fsnotify/fsnotify" 11   "github.com/kardianos/osext" 12  ) 13   14  func syscallExec() { 15   watcher, err := setupWatcher() 16   if err != nil { 17     log.Fatal(err) 18   } 19   defer finally(watcher) 20   21   fmt.Printf("current pid: %d\n", os.Getpid()) 22   var count = 0 23   24   go func(count int) { 25     for { 26       fmt.Printf(">>> count in GO ROUTINE: %d\n", count) 27       count++ 28       time.Sleep(1 * time.Second) 29     } 30   }(count) 31   32   for { 33     fmt.Printf(">>> count in MAIN: %d\n", count) 34     count++ 35     time.Sleep(1 * time.Second) 36   } 37  } 38   39  func finally(watcher chan struct{}) { 40   // 重啟時(shí)沒(méi)有執(zhí)行此函數(shù) 41   fmt.Println("exit original exec") 42   close(watcher) 43  } 44   45  func setupWatcher() (chan struct{}, error) { 46   file, err := osext.Executable() 47   if err != nil { 48     return nil, err 49   } 50   log.Printf("watching %q\n", file) 51   w, err := fsnotify.NewWatcher() 52   if err != nil { 53     return nil, err 54   } 55   done := make(chan struct{}) 56   go func() { 57     select { 58     case e := <-w.Events: 59       log.Printf("watcher received: %v", e) 60       err := syscall.Exec(file, os.Args, os.Environ()) 61       if err != nil { 62         log.Fatal(err) 63       } 64     case err := <-w.Errors: 65       log.Printf("watcher error: %+v", err) 66     case <-done: 67       log.Print("watcher shutting down") 68       return 69     } 70   }() 71   err = w.Add(file) 72   if err != nil { 73     return nil, err 74   } 75   return done, nil 76  }

本站僅提供存儲(chǔ)服務(wù),所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊舉報(bào)。
打開(kāi)APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
gRPC服務(wù)發(fā)現(xiàn)&負(fù)載均衡
使用 Casbin 在 Golang 項(xiàng)目中授權(quán)
Golang工程經(jīng)驗(yàn)
golang操作mysql使用總結(jié)
RabbitMQ Golang教程(二)
Go 語(yǔ)言文件操作示例
更多類似文章 >>
生活服務(wù)
熱點(diǎn)新聞
分享 收藏 導(dǎo)長(zhǎng)圖 關(guān)注 下載文章
綁定賬號(hào)成功
后續(xù)可登錄賬號(hào)暢享VIP特權(quán)!
如果VIP功能使用有故障,
可點(diǎn)擊這里聯(lián)系客服!

聯(lián)系客服