Go语言是使用包来组织源代码的,包(package)是多个 Go 源码的集合,是一种高级的代码复用方案
Go语言中为我们提供了很多内置包,如 fmt、os、io 等。
任何源代码文件必须属于某个包,同时源码文件的第一行有效代码必须是package pacakgeName
语句,通过该语句声明自己所在的包
Go语言的包借助了目录树的组织形式,一般包的名称就是其源文件所在目录的名称
虽然Go语言没有强制要求包名必须和其所在的目录名同名,但还是建议包名和所在目录同名,这样结构更清晰
包可以定义在很深的目录中,包名的定义是不包括目录路径的,但是包在引用时一般使用全路径引用
包的习惯用法:
-
等特殊符号要在代码中引用其他包的内容,需要使用 import 关键字导入使用的包。具体语法如下:
import "包的路径"
注意事项:
包的导入有两种写法,分别是单行导入和多行导入
//单行导入
import "包 1 的路径"
import "包 2 的路径"
//多行导入
import (
"包 1 的路径"
"包 2 的路径"
)
注意:
包的绝对路径就是GOROOT/src/
或GOPATH
后面包的存放路径
标准引用格式
package main
import "fmt"
func main() {
fmt.Println("hello world")
}
此时可以用fmt.
作为前缀来使用 fmt
包中的方法,这是常用的一种方式
自定义别名引用格式
package main
import F "fmt"
func main() {
F.Println("hello world")
}
其中 F 就是 fmt 包的别名,使用时我们可以使用F.
来代替标准引用格式的fmt.
来作为前缀使用 fmt 包中的方法
省略引用格式
package main
import . "fmt"
func main() {
//不需要加前缀 fmt.
Println("hello world")
}
这种格式相当于把 fmt 包直接合并到当前程序中,在使用 fmt 包内的方法是可以不用加前缀fmt.
,直接引用
匿名引用格式
package main
import (
_ "database/sql"
"fmt"
)
func main() {
fmt.Println("hello world")
}
在引用某个包时,如果只是希望执行包初始化的 init 函数,而不使用包内部的数据时,可以使用匿名引用格式
匿名导入的包与其他方式导入的包一样都会被编译到可执行文件中
使用标准格式引用包,但是代码中却没有使用包,编译器会报错。如果包中有 init 初始化函数,则通过import _ "包的路径"
这种方式引用包,仅执行包的初始化函数,即使包没有 init 初始化函数,也不会引发编译器报错
注意:
go module 是Go语言从 1.11 版本之后官方推出的版本管理工具,并且从 Go1.13 版本开始,go module 成为了Go语言默认的依赖管理工具
Modules 官方定义为
Modules 是相关 Go 包的集合,是源代码交换和版本控制的单元
Go语言命令直接支持使用 Modules,包括记录和解析对其他模块的依赖性,Modules 替换旧的基于 GOPATH 的方法,来指定使用哪些源文件
使用go module之前需要设置环境变量:
GO111MODULE 有三个值:off, on和auto(默认值)
go mod 有以下命令
命令 | 说明 |
---|---|
downloadx | download modules to local cache(下载依赖包) |
edit | edit go.mod from tools or scripts(编辑go.mod) |
graph | print module requirement graph (打印模块依赖图) |
init | initialize new module in current directory(在当前目录初始化mod) |
tidy | add missing and remove unused modules(拉取缺少的模块,移除不用的模块) |
vendor | make vendored copy of dependencies(将依赖复制到vendor下) |
verify | verify dependencies have expected content (验证依赖是否正确) |
why | explain why packages or modules are needed(解释为什么需要依赖) |
init tdiy edit
使用go get命令下载指定版本的依赖包:
执行go get
命令,在下载依赖包的同时还可以指定依赖包的版本。
go get -u
命令会将项目中的包升级到最新的次要版本或者修订版本;go get -u=patch
命令会将项目中的包升级到最新的修订版本;go get [包名]@[版本号]
命令会下载对应包的指定版本或者将对应包升级到指定的版本。提示:
go get [包名]@[版本号]
命令中版本号可以是 x.y.z 的形式,例如 go get [email protected],也可以是 git 上的分支或 tag,例如 go get foo@master,还可以是 git 提交时的哈希值,例如 go get foo@e3702bed2
项目中使用
在 GOPATH 目录下新建一个目录,并使用go mod init
初始化生成 go.mod 文件
go.mod 文件一旦创建后,它的内容将会被 go toolchain 全面掌控,go toolchain 会在各类命令执行时,比如go get
、go build
、go mod
等修改和维护 go.mod 文件
go.mod 提供了 module、require、replace 和 exclude 四个命令:
初始化生成的 go.mod 文件如下所示:
添加依赖
新建一个 main.go 文件,写入以下代码
package main
import (
"net/http"
"github.com/labstack/echo"
)
func main() {
e := echo.New()
e.GET("/", func(c echo.Context) error {
return c.String(http.StatusOK, "Hello, World!")
})
e.Logger.Fatal(e.Start(":1323"))
}
执行go mod tidy
运行代码会发现 go mod 会自动查找依赖自动下载
go module 安装 package 的原则是先拉取最新的 release tag,若无 tag 则拉取最新的 commit
go 会自动生成一个 go.sum 文件来记录 dependency tree。
执行脚本go run main.go
,就可以运行项目。
可以使用命令go list -m -u all
来检查可以升级的 package,使用go get -u need-upgrade-package
升级后会将新的依赖版本更新到 go.mod
比如:go get -u github.com/labstack/gommon
也可以使用go get -u
升级所有依赖。
使用 replace 替换无法直接获取的 package:
由于某些已知的原因,并不是所有的 package 都能成功下载,比如:golang.org 下的包。
modules 可以通过在 go.mod 文件中使用 replace 指令替换成 github 上对应的库,比如:golang.org 下的包
replace (
golang.org/x/crypto v0.0.0-20190313024323-a1f597ede03a => github.com/golang/crypto v0.0.0-20190313024323-a1f597ede03a
)
modules 可以通过在 go.mod 文件中使用 replace 指令替换成 github 上对应的库,比如:
replace (
golang.org/x/crypto v0.0.0-20190313024323-a1f597ede03a => github.com/golang/crypto v0.0.0-20190313024323-a1f597ede03a
)
go install
命令将项目打包安装为可执行文件,在安装在GOPATH的bin目录下,go install执行的项目 必须有main方法