上一篇文章裏咱們介紹了go modules的初步使用,如今咱們來更深刻的瞭解一下如何使用go get在module中管理依賴。html
首先咱們介紹過go mod edit修改go.mod,然而它有兩點缺陷:git
好消息是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呢?
這就是版本選擇的內容了。
之前有過gopkg.in+go get這種解決方案,而新的go get所支持的版本選擇則是這一方案的進一步擴展,看幾條規則:
那麼咱們想要把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的資料還不完善,因此我也是對着文檔邊看邊作,不免會有疏漏和錯誤,歡迎你們指正!