在go modules裏使用go get進行包管理

上一篇文章裏咱們介紹了go modules的初步使用,如今咱們來更深刻的瞭解一下如何使用go get在module中管理依賴。html

 

module下的包管理

首先咱們介紹過go mod edit修改go.mod,然而它有兩點缺陷:git

  • 首先是它的-require必須接受「package@version」這種形式,缺一不可,並且不能識別文檔規定的master和latest標誌。
  • 其次是edit只適合用於修改依賴版本,給package更名,屏蔽特定的package這三個功能,不適用於添加依賴。

好消息是go get如今有了在modules中添加/修改/更新package的能力。github

想要完總體驗go modules,咱們須要選擇一個GOPATH之外的目錄,而且設置GO11MODULE=on,這樣使用go get時只會影響當前的main module,不會污染GOPATH。golang

此次我選用本身作着玩的玩具項目演示在沒有進行包管理的項目中使用go modules。(關於vendor的遷移,可使用go mod vendor -v,詳細介紹之後會有)。chrome

咱們將項目clone到非GOPATH的路徑下,而後使用數據庫

go mod init [project name]

初始化module。初始化後的目錄:工具

這時go.mod仍是空的,咱們知道go build會更新go.mod,因此咱們先go buildui

默認會使用go get得到latest的package,如今go.mod已經被更新了,項目也成功被編譯,這是go.mod:spa

module schanclient require ( github.com/PuerkitoBio/goquery v1.4.1 github.com/andybalholm/cascadia v1.0.0 // indirect
    github.com/chromedp/chromedp v0.1.2 golang.org/x/net v0.0.0-20180826012351-8a410e7b638d // indirect
)

indirect的意思是指這個package被子module/package依賴了,可是main module並無直接import使用,也就是所謂的間接引用。code

一般,go.mod使用默認行爲就能夠很好地完成包管理,不過生活中老是有些例外。

咱們看到chromedp使用了0.1.2版本,這是三個月前的版本了,最新的commit在上個月,go mod edit須要明確指定版本號或者commit的時間+checksum,顯然這很麻煩,不是咱們但願的。

那麼咱們要如何才能使用最新的版本而不是最新的tags呢?

又或者咱們不想要最新的版本,須要某個特定版本的package呢?

這就是版本選擇的內容了。

 

go get的新特性——版本選擇

之前有過gopkg.in+go get這種解決方案,而新的go get所支持的版本選擇則是這一方案的進一步擴展,看幾條規則:

  • go get會自動下載並安裝package,而後更新到go.mod中
  • 可使用go get package[@version]來安裝指定版本的package,不指定version時默認行爲和go get package@latest同樣
  • version能夠是vx.y.z這種形式或者直接使用commit的checksum,也能夠是master或者latest
  • 當version是latest時,也就是默認行爲,對於有tags的package,會選取最新的tag,對於沒有tags的package,則選取最新的commit
  • 當version是master時,無論package有沒有打tag,都會選擇master分支的最新commit
  • 能夠在version前使用>,>=,<,<=,表示選取的版本不得超過/低於version,在這個範圍內的符合latest條件的版本
  • go get -u能夠更新package到latest版本
  • go get -u=patch將只更新小版本,例如從v1.2.4到v1.2.5
  • 當想要修改package的版本時,只須要go get package@指定的version便可

那麼咱們想要把chromedp改用最新版本就很簡單了:

go get github.com/chromedp/chromedp@master

如今go.mod裏已經將chromedp更新了:

module schanclient

require (
	github.com/PuerkitoBio/goquery v1.4.1
	github.com/andybalholm/cascadia v1.0.0 // indirect
	github.com/chromedp/chromedp v0.1.3-0.20180717231922-bf52fed0d3e6
	golang.org/x/net v0.0.0-20180826012351-8a410e7b638d // indirect
)

若是咱們如今想要添加額外的package呢?

直接使用go get就能夠了,好比我如今想用gorm往數據庫存數據:

go get github.com/jinzhu/gorm

更新後的go.mod

module schanclient

require (
	github.com/PuerkitoBio/goquery v1.4.1
	github.com/andybalholm/cascadia v1.0.0 // indirect
	github.com/chromedp/chromedp v0.1.3-0.20180717231922-bf52fed0d3e6
	github.com/jinzhu/gorm v1.9.1 // indirect
	github.com/jinzhu/inflection v0.0.0-20180308033659-04140366298a // indirect
	golang.org/x/net v0.0.0-20180826012351-8a410e7b638d // indirect
)

咱們看到latest版本的gorm已經被添加了,固然由於咱們在main module裏沒有import使用它,因此是indirect的。

若是咱們想用v1.9的gorm:

go get github.com/jinzhu/gorm@v1.9

很遺憾,版本選擇是從大版本到小版本的順序,若是有v1.9和v1.9.1,那麼當你指定v1.9時會自動選取小版本號最高的版本,除非除了v1.9以外沒有其餘的v1.9.z的tag存在,在這裏就是v1.9.1

 

還有一點值得一提,go build和go test只會將go.mod中沒有的package添加進去,不會覆蓋或者改變go get引入的規則,因此不用擔憂他們會衝突

 

是否是以爲和venv+pip很像,沒錯,這說明go的包管理工具也逐漸步入現代化了。

至於屏蔽package,刪除package以及爲package更名(好比golang.org/x/...的訪問不了的package),這些是go mod edit的功能,具體的請查看go help mod edit。

 

由於go modules的資料還不完善,因此我也是對着文檔邊看邊作,不免會有疏漏和錯誤,歡迎你們指正!

相關文章
相關標籤/搜索