Go 命令

2022-05-13 17:55 更新

Go 命令

Go語(yǔ)言自帶有一套完整的命令操作工具,你可以通過(guò)在命令行中執(zhí)行go來(lái)查看它們:


圖1.3 Go命令顯示詳細(xì)的信息

這些命令對(duì)于我們平時(shí)編寫(xiě)的代碼非常有用,接下來(lái)就讓我們了解一些常用的命令。

go build

這個(gè)命令主要用于編譯代碼。在包的編譯過(guò)程中,若有必要,會(huì)同時(shí)編譯與之相關(guān)聯(lián)的包。

  • 如果是普通包,就像我們?cè)?.2節(jié)中編寫(xiě)的mymath包那樣,當(dāng)你執(zhí)行g(shù)o build之后,它不會(huì)產(chǎn)生任何文件。如果你需要在$GOPATH/pkg下生成相應(yīng)的文件,那就得執(zhí)行g(shù)o install。
  • 如果是main包,當(dāng)你執(zhí)行g(shù)o build之后,它就會(huì)在當(dāng)前目錄下生成一個(gè)可執(zhí)行文件。如果你需要在$GOPATH/bin下生成相應(yīng)的文件,需要執(zhí)行g(shù)o install,或者使用go build -o 路徑/a.exe。
  • 如果某個(gè)項(xiàng)目文件夾下有多個(gè)文件,而你只想編譯某個(gè)文件,就可在go build之后加上文件名,例如go build a.go;go build命令默認(rèn)會(huì)編譯當(dāng)前目錄下的所有g(shù)o文件。
  • 你也可以指定編譯輸出的文件名。例如1.2節(jié)中的mathapp應(yīng)用,我們可以指定go build -o astaxie.exe,默認(rèn)情況是你的package名(非main包),或者是第一個(gè)源文件的文件名(main包)。

(注:實(shí)際上,package名在Go語(yǔ)言規(guī)范中指代碼中“package”后使用的名稱,此名稱可以與文件夾名不同。默認(rèn)生成的可執(zhí)行文件名是文件夾名。)

  • go build會(huì)忽略目錄下以“_”或“.”開(kāi)頭的go文件。
  • 如果你的源代碼針對(duì)不同的操作系統(tǒng)需要不同的處理,那么你可以根據(jù)不同的操作系統(tǒng)后綴來(lái)命名文件。例如有一個(gè)讀取數(shù)組的程序,它對(duì)于不同的操作系統(tǒng)可能有如下幾個(gè)源文件:array_linux.go array_darwin.go array_windows.go array_freebsd.go

go build的時(shí)候會(huì)選擇性地編譯以系統(tǒng)名結(jié)尾的文件(Linux、Darwin、Windows、Freebsd)。例如Linux系統(tǒng)下面編譯只會(huì)選擇array_linux.go文件,其它系統(tǒng)命名后綴文件全部忽略。

參數(shù)的介紹

  • -o 指定輸出的文件名,可以帶上路徑,例如 go build -o a/b/c
  • -i 安裝相應(yīng)的包,編譯+go install
  • -a 更新全部已經(jīng)是最新的包的,但是對(duì)標(biāo)準(zhǔn)包不適用
  • -n 把需要執(zhí)行的編譯命令打印出來(lái),但是不執(zhí)行,這樣就可以很容易的知道底層是如何運(yùn)行的
  • -p n 指定可以并行可運(yùn)行的編譯數(shù)目,默認(rèn)是CPU數(shù)目
  • -race 開(kāi)啟編譯的時(shí)候自動(dòng)檢測(cè)數(shù)據(jù)競(jìng)爭(zhēng)的情況,目前只支持64位的機(jī)器
  • -v 打印出來(lái)我們正在編譯的包名
  • -work 打印出來(lái)編譯時(shí)候的臨時(shí)文件夾名稱,并且如果已經(jīng)存在的話就不要?jiǎng)h除
  • -x 打印出來(lái)執(zhí)行的命令,其實(shí)就是和-n的結(jié)果類似,只是這個(gè)會(huì)執(zhí)行
  • -ccflags 'arg list' 傳遞參數(shù)給5c, 6c, 8c 調(diào)用
  • -compiler name 指定相應(yīng)的編譯器,gccgo還是gc
  • -gccgoflags 'arg list' 傳遞參數(shù)給gccgo編譯連接調(diào)用
  • -gcflags 'arg list' 傳遞參數(shù)給5g, 6g, 8g 調(diào)用
  • -installsuffix suffix 為了和默認(rèn)的安裝包區(qū)別開(kāi)來(lái),采用這個(gè)前綴來(lái)重新安裝那些依賴的包,-race的時(shí)候默認(rèn)已經(jīng)是-installsuffix race,大家可以通過(guò)-n命令來(lái)驗(yàn)證
  • -ldflags 'flag list' 傳遞參數(shù)給5l, 6l, 8l 調(diào)用
  • -tags 'tag list' 設(shè)置在編譯的時(shí)候可以適配的那些tag,詳細(xì)的tag限制參考里面的 Build Constraints

go clean

這個(gè)命令是用來(lái)移除當(dāng)前源碼包和關(guān)聯(lián)源碼包里面編譯生成的文件。這些文件包括

_obj/            舊的object目錄,由Makefiles遺留
_test/           舊的test目錄,由Makefiles遺留
_testmain.go     舊的gotest文件,由Makefiles遺留
test.out         舊的test記錄,由Makefiles遺留
build.out       舊的test記錄,由Makefiles遺留
*.[568ao]       object文件,由Makefiles遺留
?
DIR(.exe)       由go build產(chǎn)生
DIR.test(.exe)   由go test -c產(chǎn)生
MAINFILE(.exe)   由go build MAINFILE.go產(chǎn)生
*.so             由 SWIG 產(chǎn)生

我一般都是利用這個(gè)命令清除編譯文件,然后GitHub遞交源碼,在本機(jī)測(cè)試的時(shí)候這些編譯文件都是和系統(tǒng)相關(guān)的,但是對(duì)于源碼管理來(lái)說(shuō)沒(méi)必要。

$ go clean -i -n
cd /Users/astaxie/develop/gopath/src/mathapp
rm -f mathapp mathapp.exe mathapp.test mathapp.test.exe app app.exe
rm -f /Users/astaxie/develop/gopath/bin/mathapp

參數(shù)介紹

  • -i 清除關(guān)聯(lián)的安裝的包和可運(yùn)行文件,也就是通過(guò)go install安裝的文件
  • -n 把需要執(zhí)行的清除命令打印出來(lái),但是不執(zhí)行,這樣就可以很容易的知道底層是如何運(yùn)行的
  • -r 循環(huán)的清除在import中引入的包
  • -x 打印出來(lái)執(zhí)行的詳細(xì)命令,其實(shí)就是-n打印的執(zhí)行版本

go fmt

有過(guò)C/C++經(jīng)驗(yàn)的讀者會(huì)知道,一些人經(jīng)常為代碼采取K&R風(fēng)格還是ANSI風(fēng)格而爭(zhēng)論不休。在go中,代碼則有標(biāo)準(zhǔn)的風(fēng)格。由于之前已經(jīng)有的一些習(xí)慣或其它的原因我們常將代碼寫(xiě)成ANSI風(fēng)格或者其它更合適自己的格式,這將為人們?cè)陂喿x別人的代碼時(shí)添加不必要的負(fù)擔(dān),所以go強(qiáng)制了代碼格式(比如左大括號(hào)必須放在行尾),不按照此格式的代碼將不能編譯通過(guò),為了減少浪費(fèi)在排版上的時(shí)間,go工具集中提供了一個(gè)go fmt命令 它可以幫你格式化你寫(xiě)好的代碼文件,使你寫(xiě)代碼的時(shí)候不需要關(guān)心格式,你只需要在寫(xiě)完之后執(zhí)行g(shù)o fmt <文件名>.go,你的代碼就被修改成了標(biāo)準(zhǔn)格式,但是我平常很少用到這個(gè)命令,因?yàn)殚_(kāi)發(fā)工具里面一般都帶了保存時(shí)候自動(dòng)格式化功能,這個(gè)功能其實(shí)在底層就是調(diào)用了go fmt。接下來(lái)的一節(jié)我將講述兩個(gè)工具,這兩個(gè)工具都自帶了保存文件時(shí)自動(dòng)化go fmt功能。

使用go fmt命令,其實(shí)是調(diào)用了gofmt,而且需要參數(shù)-w,否則格式化結(jié)果不會(huì)寫(xiě)入文件。gofmt -w -l src,可以格式化整個(gè)項(xiàng)目。

所以go fmt是gofmt的上層一個(gè)包裝的命令,我們想要更多的個(gè)性化的格式化可以參考 gofmt

gofmt的參數(shù)介紹

  • -l 顯示那些需要格式化的文件
  • -w 把改寫(xiě)后的內(nèi)容直接寫(xiě)入到文件中,而不是作為結(jié)果打印到標(biāo)準(zhǔn)輸出。
  • -r 添加形如“a[b:len(a)] -> a[b:]”的重寫(xiě)規(guī)則,方便我們做批量替換
  • -s 簡(jiǎn)化文件中的代碼
  • -d 顯示格式化前后的diff而不是寫(xiě)入文件,默認(rèn)是false
  • -e 打印所有的語(yǔ)法錯(cuò)誤到標(biāo)準(zhǔn)輸出。如果不使用此標(biāo)記,則只會(huì)打印不同行的前10個(gè)錯(cuò)誤。
  • -cpuprofile 支持調(diào)試模式,寫(xiě)入相應(yīng)的cpufile到指定的文件

go get

這個(gè)命令是用來(lái)動(dòng)態(tài)獲取遠(yuǎn)程代碼包的,目前支持的有BitBucket、GitHub、Google Code和Launchpad。這個(gè)命令在內(nèi)部實(shí)際上分成了兩步操作:第一步是下載源碼包,第二步是執(zhí)行g(shù)o install。下載源碼包的go工具會(huì)自動(dòng)根據(jù)不同的域名調(diào)用不同的源碼工具,對(duì)應(yīng)關(guān)系如下:

BitBucket (Mercurial Git)
GitHub (Git)
Google Code Project Hosting (Git, Mercurial, Subversion)
Launchpad (Bazaar)

所以為了go get 能正常工作,你必須確保安裝了合適的源碼管理工具,并同時(shí)把這些命令加入你的PATH中。其實(shí)go get支持自定義域名的功能,具體參見(jiàn)go help remote。

參數(shù)介紹:

  • -d 只下載不安裝
  • -f 只有在你包含了-u參數(shù)的時(shí)候才有效,不讓-u去驗(yàn)證import中的每一個(gè)都已經(jīng)獲取了,這對(duì)于本地fork的包特別有用
  • -fix 在獲取源碼之后先運(yùn)行fix,然后再去做其他的事情
  • -t 同時(shí)也下載需要為運(yùn)行測(cè)試所需要的包
  • -u 強(qiáng)制使用網(wǎng)絡(luò)去更新包和它的依賴包
  • -v 顯示執(zhí)行的命令

go install

這個(gè)命令在內(nèi)部實(shí)際上分成了兩步操作:第一步是生成結(jié)果文件(可執(zhí)行文件或者.a包),第二步會(huì)把編譯好的結(jié)果移到$GOPATH/pkg或者$GOPATH/bin。

參數(shù)支持go build的編譯參數(shù)。大家只要記住一個(gè)參數(shù)-v就好了,這個(gè)隨時(shí)隨地的可以查看底層的執(zhí)行信息。

go test

執(zhí)行這個(gè)命令,會(huì)自動(dòng)讀取源碼目錄下面名為*_test.go的文件,生成并運(yùn)行測(cè)試用的可執(zhí)行文件。輸出的信息類似

ok   archive/tar   0.011s
FAIL archive/zip   0.022s
ok   compress/gzip 0.033s
...

默認(rèn)的情況下,不需要任何的參數(shù),它會(huì)自動(dòng)把你源碼包下面所有test文件測(cè)試完畢,當(dāng)然你也可以帶上參數(shù),詳情請(qǐng)參考go help testflag

這里我介紹幾個(gè)我們常用的參數(shù):

  • -bench regexp 執(zhí)行相應(yīng)的benchmarks,例如 -bench=.
  • -cover 開(kāi)啟測(cè)試覆蓋率
  • -run regexp 只運(yùn)行regexp匹配的函數(shù),例如 -run=Array 那么就執(zhí)行包含有Array開(kāi)頭的函數(shù)
  • -v 顯示測(cè)試的詳細(xì)命令

go tool

go tool下面下載聚集了很多命令,這里我們只介紹兩個(gè),fix和vet

  • go tool fix . 用來(lái)修復(fù)以前老版本的代碼到新版本,例如go1之前老版本的代碼轉(zhuǎn)化到go1,例如API的變化
  • go tool vet directory|files 用來(lái)分析當(dāng)前目錄的代碼是否都是正確的代碼,例如是不是調(diào)用fmt.Printf里面的參數(shù)不正確,例如函數(shù)里面提前return了然后出現(xiàn)了無(wú)用代碼之類的。

go generate

這個(gè)命令是從Go1.4開(kāi)始才設(shè)計(jì)的,用于在編譯前自動(dòng)化生成某類代碼。go generate和go build是完全不一樣的命令,通過(guò)分析源碼中特殊的注釋,然后執(zhí)行相應(yīng)的命令。這些命令都是很明確的,沒(méi)有任何的依賴在里面。而且大家在用這個(gè)之前心里面一定要有一個(gè)理念,這個(gè)go generate是給你用的,不是給使用你這個(gè)包的人用的,是方便你來(lái)生成一些代碼的。

這里我們來(lái)舉一個(gè)簡(jiǎn)單的例子,例如我們經(jīng)常會(huì)使用yacc來(lái)生成代碼,那么我們常用這樣的命令:

go tool yacc -o gopher.go -p parser gopher.y

-o 指定了輸出的文件名, -p指定了package的名稱,這是一個(gè)單獨(dú)的命令,如果我們想讓go generate來(lái)觸發(fā)這個(gè)命令,那么就可以在當(dāng)前目錄的任意一個(gè)xxx.go文件里面的任意位置增加一行如下的注釋:

//go:generate go tool yacc -o gopher.go -p parser gopher.y

這里我們注意了,//go:generate是沒(méi)有任何空格的,這其實(shí)就是一個(gè)固定的格式,在掃描源碼文件的時(shí)候就是根據(jù)這個(gè)來(lái)判斷的。

所以我們可以通過(guò)如下的命令來(lái)生成,編譯,測(cè)試。如果gopher.y文件有修改,那么就重新執(zhí)行g(shù)o generate重新生成文件就好。

$ go generate
$ go build
$ go test

godoc

在Go1.2版本之前還支持go doc命令,但是之后全部移到了godoc這個(gè)命令下,需要這樣安裝go get golang.org/x/tools/cmd/godoc

很多人說(shuō)go不需要任何的第三方文檔,例如chm手冊(cè)之類的(其實(shí)我已經(jīng)做了一個(gè)了,chm手冊(cè)),因?yàn)樗鼉?nèi)部就有一個(gè)很強(qiáng)大的文檔工具。

如何查看相應(yīng)package的文檔呢? 例如builtin包,那么執(zhí)行g(shù)odoc builtin 如果是http包,那么執(zhí)行g(shù)odoc net/http 查看某一個(gè)包里面的函數(shù),那么執(zhí)行g(shù)odoc fmt Printf 也可以查看相應(yīng)的代碼,執(zhí)行g(shù)odoc -src fmt Printf

通過(guò)命令在命令行執(zhí)行 godoc -http=:端口號(hào) 比如godoc -http=:8080。然后在瀏覽器中打開(kāi)127.0.0.1:8080,你將會(huì)看到一個(gè)golang.org的本地copy版本,通過(guò)它你可以查詢pkg文檔等其它內(nèi)容。如果你設(shè)置了GOPATH,在pkg分類下,不但會(huì)列出標(biāo)準(zhǔn)包的文檔,還會(huì)列出你本地GOPATH中所有項(xiàng)目的相關(guān)文檔,這對(duì)于經(jīng)常被墻的用戶來(lái)說(shuō)是一個(gè)不錯(cuò)的選擇。

其它命令

go還提供了其它很多的工具,例如下面的這些工具

go version 查看go當(dāng)前的版本
go env 查看當(dāng)前go的環(huán)境變量
go list 列出當(dāng)前全部安裝的package
go run 編譯并運(yùn)行Go程序

以上這些工具還有很多參數(shù)沒(méi)有一一介紹,用戶可以使用go help 命令獲取更詳細(xì)的幫助信息。


以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)