go build -tags構(gòu)建約束試驗(yàn)示例解析
使用方法
- 構(gòu)建約束以一行
+build
開始的注釋。在+build
之后列出了一些條件,在這些條件成立時(shí),該文件應(yīng)包含在編譯的包中; - 約束可以出現(xiàn)在任何源文件中,不限于go文件;
+build
必須出現(xiàn)在package
語句之前,+build
注釋之后應(yīng)要有一個(gè)空行。
// +build debug package main import "fmt" func main() { fmt.Println("Hello World!") }
語法規(guī)則
1)只允許是字母數(shù)字或_
2)多個(gè)條件之間,空格表示OR;逗號(hào)表示AND;嘆號(hào)(!)表示NOT
3)一個(gè)文件可以有多個(gè)+build,它們之間的關(guān)系是AND。如:
// +build linux darwin // +build 386 等價(jià)于 // +build (linux OR darwin) AND 386
4)預(yù)定義了一些條件:
runtime.GOOS、runtime.GOARCH、compiler(gc或gccgo)、cgo、context.BuildTags中的其他單詞
5)如果一個(gè)文件名(不含后綴),以 *_GOOS, *_GOARCH, 或 *_GOOS_GOARCH結(jié)尾,它們隱式包含了 構(gòu)建約束
6)當(dāng)不想編譯某個(gè)文件時(shí),可以加上// +build ignore
。這里的ignore
可以是其他單詞,只是ignore
更能讓人知道什么意思
更多詳細(xì)信息,可以查看go/build/build.go
文件中shouldBuild
和match
方法。
應(yīng)用實(shí)例 Build constraints來實(shí)現(xiàn)
除了*_GOOS這種預(yù)定義的應(yīng)用,我們看一個(gè)實(shí)際的應(yīng)用。
比如,項(xiàng)目中需要在測試環(huán)境輸出Debug信息,一般通過一個(gè)變量(或常量)來控制是測試環(huán)境還是生產(chǎn)環(huán)境,比如:if DEBUG {},這樣在生產(chǎn)環(huán)境每次也會(huì)進(jìn)行這樣的判斷。在golang-nuts郵件列表中有人問過這樣的問題,貌似沒有討論出更好的方法(想要跟C中條件編譯一樣的效果)。下面我們采用Build constraints來實(shí)現(xiàn)。
1)文件列表:main.go logger_debug.go logger_product.go
2)在main.go中簡單的調(diào)用Debug()方法。
3)在logger_product.go中的Debug()是空實(shí)現(xiàn),但是在文件開始加上// + build !debug
4)在logger_debug.go中的Debug()是需要輸出的調(diào)試信息,同時(shí)在文件開始加上// + build debug
這樣,在測試環(huán)境編譯的時(shí)傳遞-tags參數(shù):go build/install -tags “debug” logger。生產(chǎn)環(huán)境:go build/install logger就行了。
對于生產(chǎn)環(huán)境,不傳遞-tags時(shí),為什么會(huì)編譯logger_product.go呢?因?yàn)樵趃o/build/build.go中的match方法中有這么一句:
if strings.HasPrefix(name, "!") { // negation return len(name) > 1 && !ctxt.match(name[1:]) }
也就是說,只要有!(不能只是!),tag不在BuildTags中時(shí),總是會(huì)編譯。
應(yīng)用實(shí)例 編譯的tag差異
本例程中,編譯的tag差異較大,兩個(gè)文件中一個(gè)是hash tag,一個(gè)是int tag,需要引入第三個(gè)tag來區(qū)分編譯的文件。否則,只要不帶!的tag都會(huì)被編譯進(jìn)包。
display_hash.go
// +build hash !display_alternatives // 上面 package main import "fmt" type DisplayName string func Print(name DisplayName) { fmt.Printf("%s\n", name) } func MakeDisplayName(name string) DisplayName { return DisplayName(name) }
display_int.go
// +build int package main import ( "fmt" "encoding/hex" "encoding/binary" ) type DisplayName uint64 func Print(name DisplayName) { fmt.Printf("%d\n", name) } func MakeDisplayName(name string) DisplayName { h, err := hex.DecodeString(name) if err != nil { panic(fmt.Sprintf("decode hex string failed. cause: %v\n", err)) } fmt.Printf("data: %v\n", h) value := binary.BigEndian.Uint16(h) return DisplayName(value) }
build_tag.go
package main import ( "sync" "fmt" "math/rand" "time" ) func main() { var name DisplayName name = MakeDisplayName("FAD9C812") Print(name)
編譯display_int.go
編譯執(zhí)行過程 go build -tags "display_alternatives int"
編譯display_hash.go
編譯執(zhí)行過程 go build -tags hash
以上就是go build -tags構(gòu)建約束試驗(yàn)示例解析的詳細(xì)內(nèi)容,更多關(guān)于go build tags約束構(gòu)建的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
詳解Go多協(xié)程并發(fā)環(huán)境下的錯(cuò)誤處理
這篇文章主要介紹了詳解Go多協(xié)程并發(fā)環(huán)境下的錯(cuò)誤處理,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08GoFrame框架gset使用對比PHP?Java?Redis優(yōu)勢
這篇文章主要為大家介紹了GoFrame框架gset對比PHP?Java?Redis的使用優(yōu)勢詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06Go集成swagger實(shí)現(xiàn)在線接口文檔的教程指南
wagger是一個(gè)用于設(shè)計(jì),構(gòu)建和文檔化API的開源框架,在Go語言中,Swagger可以幫助后端開發(fā)人員快速創(chuàng)建和定義RESTful API,并提供自動(dòng)生成接口文檔的功能,所以本文給大家介紹了Go集成swagger實(shí)現(xiàn)在線接口文檔的方法,需要的朋友可以參考下2024-11-11go語言定時(shí)器Timer及Ticker的功能使用示例詳解
這篇文章主要為大家介紹了go語言定時(shí)器的功能使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2022-04-04