欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

詳解Golang如何在編譯時(shí)注入版本信息

 更新時(shí)間:2023年06月18日 08:58:07   作者:Shawn27  
這篇文章主要為大家詳細(xì)介紹了Golang如何在編譯時(shí)實(shí)現(xiàn)注入版本信息,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的可以了解一下

問題

一般而言,稍微做得好一點(diǎn)的開源軟件,其二進(jìn)制文件都會(huì)帶上版本信息,比如Docker,你可以通過以下命令來查看:

$ docker --version
Docker version 20.10.17, build 100c701
$ docker version
...
//輸出太長,這里就不展示了

有些Web應(yīng)用還提供了/version接口,你可以通過接口來獲取:

$ curl http://127.0.0.1:8080/version | jq
{
  "Build Time": "Fri, 16 Jun 2023 16:19:23 +0800",
  "Git Commit": "b554659",
  "Go Version": "go1.19",
  "OS/Arch": "darwin/amd64",
  "version": "0.9.2"
}

然而很多公司并沒有這方面的強(qiáng)制性規(guī)范,多數(shù)開發(fā)人員也沒有這樣的意識(shí)或習(xí)慣,所以大部分項(xiàng)目并沒有實(shí)現(xiàn)這個(gè)功能。如果二進(jìn)制文件不帶上版本信息,你可能要借助其它系統(tǒng)來查詢或追溯,這無疑是一件難受的事,對(duì)于代碼調(diào)試和問題定位很不友好。

那么在Golang中,如何實(shí)現(xiàn)這個(gè)功能呢?

思路

版本信息有很多,比如代碼版本號(hào)、Git提交號(hào)、編譯時(shí)間、Go版本號(hào)等,如何讓二進(jìn)制文件帶上這些信息呢?

硬編碼在代碼里肯定不行。因?yàn)镚it提交號(hào)是將代碼提交到代碼倉庫時(shí)才產(chǎn)生,而Go版本號(hào)和編譯時(shí)間只有在編譯時(shí)才能拿到。

比較合理的做法是在編譯時(shí)注入。Go編譯工具提供了-ldflags選項(xiàng),通過-X參數(shù)可以注入包變量的值,我們只需要在代碼中將版本信息定義成包變量,然后在編譯時(shí)完成注入即可。這個(gè)過程可以實(shí)現(xiàn)為自動(dòng)化完成,下面以make工具為例說明。

實(shí)現(xiàn)步驟

1. 在代碼中定義版本信息的變量

將代碼版本號(hào)、Git提交號(hào)、編譯時(shí)間定義為包變量:

package config
var (
	Version   string  //代碼版本號(hào)
	GitCommit string  //Git提交號(hào)
	BuildTime string  //編譯時(shí)間
)

Go版本號(hào)等信息可以借助runtime包獲取:

runtime.Version()  //Go版本
runtime.GOOS       //操作系統(tǒng)
runtime.GOARCH     //平臺(tái)架構(gòu)

2. 在代碼中使用這些變量

1)實(shí)現(xiàn)--version參數(shù)或version子命令

Cobra命令行框架為例,為rootCmd實(shí)現(xiàn)--version參數(shù):

rootCmd.Version = config.Version
rootCmd.SetVersionTemplate(fmt.Sprintf(`{{with .Name}}{{printf "%%s version information: " .}}{{end}}
    {{printf "Version:    %%s" .Version}}
    Git Commit: %s
    Build Time: %s
    Go version: %s
    OS/Arch:    %s/%s
`, config.GitCommit, config.BuildTime, runtime.Version(), runtime.GOOS, runtime.GOARCH))

2)實(shí)現(xiàn)/version接口

Gin框架為例,將/version路由到如下的Version函數(shù):

func Version(ctx *gin.Context) {
    ctx.Writer.Header().Set("Content-Type", "application/json")
    ctx.Writer.WriteHeader(http.StatusOK)
    json.NewEncoder(ctx.Writer).Encode(map[string]string{
        "version":    config.Version,
        "Git Commit": config.GitCommit,
        "Build Time": config.BuildTime,
        "Go Version": runtime.Version(),
        "OS/Arch":    runtime.GOOS + "/" + runtime.GOARCH,
    })
}

3. 在編譯前獲取版本信息

在Makefile中,用git命令獲取代碼版本號(hào)、Git提交號(hào),用date命令獲取當(dāng)前時(shí)間

VERSION    = $(shell git describe --tags --always)
GIT_COMMIT = $(shell git rev-parse --short HEAD)
BUILD_TIME = $(shell date -R)

4. 在編譯時(shí)注入版本信息

在Makefile中,通過-ldflags選項(xiàng)為go build命令傳入編譯參數(shù),實(shí)現(xiàn)版本信息的注入

define LDFLAGS
"-X 'github.com/myname/myapp/config.Version=${VERSION}' \
-X 'github.com/myname/myapp/config.GitCommit=${GIT_COMMIT}' \
-X 'github.com/myname/myapp/config.BuildTime=${BUILD_TIME}'"
endef
build:
    go build -ldflags ${LDFLAGS} -o myapp_${VERSION} .

5. 驗(yàn)證效果

編譯出二進(jìn)制,即可驗(yàn)證--version參數(shù);用二進(jìn)制啟動(dòng)服務(wù)進(jìn)程,即可驗(yàn)證/version接口。

$ make build
...
//編譯過程,省略輸出
$ myapp --version
myapp version information: 
    Version:    0.9.2
    Git Commit: b554659
    Build Time: Sat, 17 Jun 2023 11:30:44 +0800
    Go version: go1.20.5
    OS/Arch:    darwin/arm64
$ myapp
...
//啟動(dòng)進(jìn)程,省略輸出
$ curl http://127.0.0.1:8080/version | jq
{
  "Build Time": "Sat, 17 Jun 2023 11:30:44 +0800",
  "Git Commit": "b554659",
  "Go Version": "go1.20.5",
  "OS/Arch": "darwin/arm64",
  "version": "0.9.2"
}

總結(jié)

二進(jìn)制文件直接帶上版本信息,能為代碼調(diào)試和問題定位帶來很大的方便。本文介紹了通過-ldflags選項(xiàng)實(shí)現(xiàn)在編譯時(shí)注入版本信息的具體做法。

這是一個(gè)比較通用的方法,很多開源軟件比如 Docker 也是這樣做的。

到此這篇關(guān)于詳解Golang如何在編譯時(shí)注入版本信息的文章就介紹到這了,更多相關(guān)Golang編譯注入版本信息內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • go語言中if語句用法實(shí)例

    go語言中if語句用法實(shí)例

    這篇文章主要介紹了go語言中if語句用法,以實(shí)例形式分析了if語句的定義及使用技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2015-02-02
  • 淺析Go語言中閉包的使用

    淺析Go語言中閉包的使用

    閉包是一個(gè)函數(shù)和其相關(guān)的引用環(huán)境組合的一個(gè)整體。本文主要為大家介紹一下Go語言中閉包的使用,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語言有一定幫助,需要的可以參考一下
    2022-12-12
  • 使用Golang實(shí)現(xiàn)WebSocket心跳機(jī)制

    使用Golang實(shí)現(xiàn)WebSocket心跳機(jī)制

    WebSocket是一種在客戶端和服務(wù)器之間實(shí)現(xiàn)全雙工通信的協(xié)議,它允許實(shí)時(shí)地傳輸數(shù)據(jù),并且比傳統(tǒng)的HTTP請(qǐng)求更加高效,在使用Golang構(gòu)建WebSocket應(yīng)用程序時(shí),一個(gè)重要的考慮因素是如何實(shí)現(xiàn)心跳機(jī)制,所以本文將探討如何使用Golang實(shí)現(xiàn)WebSocket心跳
    2023-11-11
  • Go語言中不可不知的語法糖盤點(diǎn)

    Go語言中不可不知的語法糖盤點(diǎn)

    Go?語言有一些非常實(shí)用的語法糖(syntactic?sugar),它們使得代碼更加簡潔和易讀,本文為大家整理了15個(gè)常見的語法糖,有需要的可以了解下
    2025-01-01
  • GOLANG版的冒泡排序和快速排序分享

    GOLANG版的冒泡排序和快速排序分享

    這篇文章主要介紹了GOLANG版的冒泡排序和快速排序分享,需要的朋友可以參考下
    2015-03-03
  • idea搭建go環(huán)境實(shí)現(xiàn)go語言開發(fā)

    idea搭建go環(huán)境實(shí)現(xiàn)go語言開發(fā)

    這篇文章主要給大家介紹了關(guān)于idea搭建go環(huán)境實(shí)現(xiàn)go語言開發(fā)的相關(guān)資料,文中通過圖文介紹以及代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用go具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2024-01-01
  • Go語言對(duì)接微信支付與退款指南(示例詳解)

    Go語言對(duì)接微信支付與退款指南(示例詳解)

    在互聯(lián)網(wǎng)技術(shù)日益發(fā)展的背景下,Go語言憑借并發(fā)處理能力,在后端開發(fā)中大放異彩,本文詳細(xì)介紹如何使用Go語言對(duì)接微信支付,完成支付和退款功能,包括準(zhǔn)備工作、初始化微信支付客戶端、實(shí)現(xiàn)支付功能,以及處理支付回調(diào)和退款等
    2024-10-10
  • Go語言defer的一些神奇規(guī)則示例詳解

    Go語言defer的一些神奇規(guī)則示例詳解

    這篇文章主要為大家介紹了Go語言defer的一些神奇規(guī)則示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-11-11
  • go語言中使用timer的常用方式

    go語言中使用timer的常用方式

    這篇文章主要介紹了go語言中使用timer的常用方式,實(shí)例分析了三種常用的使用timer的方法,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-03-03
  • 基于golang的簡單分布式延時(shí)隊(duì)列服務(wù)的實(shí)現(xiàn)

    基于golang的簡單分布式延時(shí)隊(duì)列服務(wù)的實(shí)現(xiàn)

    這篇文章主要介紹了基于golang的簡單分布式延時(shí)隊(duì)列服務(wù)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02

最新評(píng)論