参考连接:https://blog.csdn.net/weixin_39003229/article/details/97638573git
Go.mod是Golang1.11版本新引入的官方包管理工具用于解决以前没有地方记录依赖包具体版本的问题,方便依赖包的管理。github
Modules是相关Go包的集合,是源代码交换和版本控制的单元。go命令直接支持使用Modules,包括记录和解析对其余模块的依赖性。Modules替换旧的基于GOPATH的方法,来指定使用哪些源文件。golang
Modules和传统的GOPATH不一样,不须要包含例如src,bin这样的子目录,一个源代码目录甚至是空目录均可以做为Modules,只要其中包含有go.mod文件。工具
如何使用go.mod?
1 首先将go的版本升级为1.11以上ui
2 设置GO111MODULE加密
GO111MODULE
GO111MODULE有三个值:off, on和auto(默认值)。 GO111MODULE=off,go命令行将不会支持module功能,寻找依赖包的方式将会沿用旧版本那种经过vendor目录或者GOPATH模式来查找。 GO111MODULE=on,go命令行会使用modules,而一点也不会去GOPATH目录下查找。 GO111MODULE=auto,默认值,go命令行将会根据当前目录来决定是否启用module功能。这种状况下能够分为两种情形: 当前目录在GOPATH/src以外且该目录包含go.mod文件 当前文件在包含go.mod文件的目录下面。
使用以前咱们要进行前期的准备,设置go的环境配置spa
go env -w GO111MODULE=on
go env -w GOPROXY=https://goproxy.cn,direct.net
1.首先咱们要在GOPATH/src 目录以外新建工程,或将老工程copy到GOPATH/src 目录以外。命令行
PS:go.mod文件一旦建立后,它的内容将会被go toolchain全面掌控。go toolchain会在各种命令执行时,好比go get、go build、go mod等修改和维护go.mod文件。版本控制
1 module语句指定包的名字(路径)
2 require语句指定的依赖项模块
3 replace语句能够替换依赖项模块
4 exclude语句能够忽略依赖项模块
下面是咱们创建了一个hello.go的文件
package main import ( "fmt" ) func main() { fmt.Println("Hello, world!") }
2.在当前目录下,命令行运行 go mod init + 模块名称 初始化模块
go mod init hello
运行完以后,会在当前目录下生成一个go.mod文件,这是一个关键文件,以后的包的管理都是经过这个文件管理。
官方说明:除了go.mod以外,go命令还维护一个名为go.sum的文件,其中包含特定模块版本内容的预期加密哈希
go命令使用go.sum文件确保这些模块的将来下载检索与第一次下载相同的位,以确保项目所依赖的模块不会出现意外更改,不管是出于恶意、意外仍是其余缘由。 go.mod和go.sum都应检入版本控制。
go.sum 不须要手工维护,因此能够不用太关注。
注意:子目录里是不须要init的,全部的子目录里的依赖都会组织在根目录的go.mod文件里
接下来,让咱们的项目依稀一下第三方包
如修改hello.go文件以下,按照过去的作法,要运行hello.go须要执行go get 命令 下载gorose包到 $GOPATH/src
package main import ( "fmt" "github.com/gohouse/gorose" ) func main() { fmt.Println("Hello, world!") }
可是,使用了新的包管理就不在须要这样作了
go run hello.go
稍等片刻… go 会自动查找代码中的包,下载依赖包,而且把具体的依赖关系和版本写入到go.mod和go.sum文件中。
查看go.mod,它会变成这样:
module test require ( github.com/gohouse/gorose v1.0.5 )
require 关键字是引用,后面是包,最后v1.11.1 是引用的版本号
这样,一个使用Go包管理方式建立项目的小例子就完成了。
问题一:依赖的包下载到哪里了?还在GOPATH里吗?
不在。
使用Go的包管理方式,依赖的第三方包被下载到了$GOPATH/pkg/mod路径下。
问题二: 依赖包的版本是怎么控制的?
在上一个问题里,能够看到最终下载在GOPATH/pkg/mod里能够保存相同包的不一样版本。
版本是在go.mod中指定的。若是,在go.mod中没有指定,go命令会自动下载代码中的依赖的最新版本,本例就是自动下载最新的版本。若是,在go.mod用require语句指定包和版本 ,go命令会根据指定的路径和版本下载包,
指定版本时能够用latest,这样它会自动下载指定包的最新版本;
问题三: 能够把项目放在$GOPATH/src下吗?
能够。可是go会根据GO111MODULE的值而采起不一样的处理方式,默认状况下,GO111MODULE=auto 自动模式
1.auto 自动模式下,项目在GOPATH/src的依赖包,在$GOPATH/src外,就使用go.mod 里 require的包
2.on 开启模式,1.12后,不管在$GOPATH/src里仍是在外面,都会使用go.mod 里 require的包
3.off 关闭模式,就是老规矩。
问题三: 依赖包中的地址失效了怎么办?好比 golang.org/x/… 下的包都没法下载怎么办?
在go快速发展的过程当中,有一些依赖包地址变动了。之前的作法:
1.修改源码,用新路径替换import的地址
2.git clone 或 go get 新包后,copy到$GOPATH/src里旧的路径下
不管什么方法,都不便于维护,特别是多人协同开发时。
使用go.mod就简单了,在go.mod文件里用 replace 替换包,例如
replace golang.org/x/text => github.com/golang/text latest
这样,go会用 github.com/golang/text 替代golang.org/x/text,原理就是下载github.com/golang/text 的最新版本到 $GOPATH/pkg/mod/golang.org/x/text下。
问题四: init生成的go.mod的模块名称有什么用?
本例里,用 go mod init hello 生成的go.mod文件里的第一行会申明module hello
由于咱们的项目已经不在$GOPATH/src里了,那么引用本身怎么办?就用模块名+路径。
例如,在项目下新建目录 utils,建立一个tools.go文件:
package utils import “fmt” func PrintText(text string) { fmt.Println(text) }
在根目录下的hello.go文件就能够 import “hello/utils” 引用utils
package main import ( "hello/utils" "github.com/astaxie/beego" ) func main() { utils.PrintText("Hi") beego.Run() }