您现在的位置是:亿华云 > IT科技类资讯
Go 项目中代码组织的两种模式
亿华云2025-10-03 06:55:32【IT科技类资讯】8人已围观
简介这是一篇基础文章,主要帮新手解决 GOPATH 和 Go Module 的问题。希望这篇文章能够为你彻底解惑。本文作者:Kade。本文的行文风格跟普通的文章不一样,是一种沉浸式的、笔记式的、或者视频稿
这是项目一篇基础文章,主要帮新手解决 GOPATH 和 Go Module 的中代织问题。希望这篇文章能够为你彻底解惑。码组模式本文作者:Kade。两种
本文的项目行文风格跟普通的文章不一样,是中代织一种沉浸式的、笔记式的码组模式、或者视频稿的两种风格。不知道你是项目否会喜欢。
注: 本文基于 go1.16,中代织 macOS 环境
01 相关概念梳理
注: 详细的可以完整阅读 Go 官方文档及 Wiki, 为了通俗一点, 文中某些描述可能不是很严谨!
首先需要清楚 Go 项目中包(package)和模块(module)的概念, 简单描述一下:
包(package)是用来管理 .go 文件的, 相关概念: 包目录, 包名, 包路径/包导入路径/导入路径它是源代码的集合, 由一个或多个源文件组成: 一个目录最多只能有一个包, 一个包只能存在于一个目录
模块(module)是用来管理包的, 相关概念: 模块目录, 模块路径它是包的集合, 由零个或多个包组成: 一个目录最多只能有一个模块, 一个模块只能存在于一个目录 ; 一个模块目录里必须要有go.mod文件
02 代码组织的两种模式
注: 文中描述模式时使用小写(强迫症); 相关的发展历史可在社区了解
GOPATH mode(gopath模式): 通过配置 GO111MODULE=off 强制开启 $GOPATH默认为用户家目录下的go目录, 即 ~/go $GOPATH可以设置多个目录, 可以实现依赖包存放在一个目录, 自己项目的包存放在另外一个目录 包需要存放在$GOPATH/src下的高防服务器子目录中, 包目录相对于$GOPATH/src的相对路径则为包的导入路径 习惯上, 包所在的目录名与包名相同(不是必须) 使用go get下载的包也是存放在$GOPATH/src目录中 依赖包可以放在vendor目录中 没有模块相关的概念 module mode(gomod模式): 通过配置GO111MODULE=on强制开启 $GOPATH默认为用户家目录下的go目录, 即 ~/go 模块目录可以是任何目录, 包必须在某个模块中 模块路径需要在模块目录下的 go.mod 文件中使用module指令指定 习惯上, 模块下的包所在的目录名与包名相同(不是必须) 使用go get下载的包存放在$GOPATH/pkg/mod下的相关目录中 通过 go 命令的参数-mod=vendor可以支持 main 包下的vendor目录 有模块相关的概念及配置, 比如: GOPROXY, GOPRIVATE, GOSUMDB等注: GO111MODULE配置还有一个值是auto, 意思是具体 go 使用哪一种模式由 go 来判断并决定, 不同版本的判断不同, 效果不同, 所有建议使用 go 之前先明确设置GO111MODULE的值为 off 或者 on
注: gomod 模式中只保留了部分的 vendor 特性支持, 不建议日常开发中使用, 一般用作依赖存档或 CI/CD 使用
注: gopath 模式基本废弃, 不建议再使用, 如果有老项目仍在使用, 建议着手迁移到 gomod 模式, 如果迁移有问题, 可以在社区交流讨论, 或向官方求助
03 两种模式的使用示例
gopath 模式(官方已经准备废弃,不建议使用)
1.开启gopath模式,码组模式 设置GO111MODULE值为off
MacBook$ # 1. 设置 MacBook$ export GO111MODULE=off MacBook$ # 需要永久配置的话, 需要修改相关的配置文件 MacBook$ # 比如: ~/.bash_profile 或 ~/.bashrc 等 MacBook$ # MacBook$ # 建议使用下面的方法: MacBook$ go env -w GO111MODULE=off MacBook$ # 2. 验证 MacBook$ go env GO111MODULE off MacBook$2.根据需要设置GOPATH, 默认值为~/go, 建议使用默认(这里为了演示设置了其他目录)
MacBook$ # 1. 设置 MacBook$ export GOPATH=/Users/kadefor/examples/gopath_mode MacBook$ # 同上建议: MacBook$ go env -w GOPATH=/Users/kadefor/examples/gopath_mode MacBook$ # 2. 验证 MacBook$ go env GOPATH /Users/kadefor/examples/gopath_mode MacBook$3.日常开发(使用labstack/echo这个 web 开发框架为例)
MacBook$ go env GO111MODULE off MacBook$ go env GOPATH /Users/kadefor/examples/gopath_mode MacBook$ pwd /Users/kadefor/examples/gopath_mode/src MacBook$ tree . . └── github.com └── myrepo └── helloworld └── main.go 3 directories, 1 file MacBook$ cd github.com/myrepo/helloworld/ MacBook$ MacBook$ # 项目代码放在`GOPATH/src`下, 一般是放在某个子目录里 MacBook$ # 相对于`GOPATH/src`的相对目录路径即为包导入路径 MacBook$ # 比如说, 有一个包cc在src目录下的`aa/bb/cc`目录里 MacBook$ # 那它的导入路径就是亿华云"aa/bb/cc" MacBook$ MacBook$ # 这里github.com/myrepo/helloworld目录下有个main包: MacBook$ pwd /Users/kadefor/examples/gopath_mode/src/github.com/myrepo/helloworld MacBook$ ls main.go MacBook$ head -8 main.go package main import ( "github.com/labstack/echo" "github.com/labstack/echo/middleware" "net/http" ) MacBook$ # gopath模式下, go找包会在`GOROOT`, `GOPATH/src`, vendor目录下去找 MacBook$ # 比如这里导入的"github.com/labstack/echo" MacBook$ # 运行看看: MacBook$ go run . main.go:4:2: cannot find package "github.com/labstack/echo" in any of: /Users/kadefor/sdk/go/src/github.com/labstack/echo (from $GOROOT) /Users/kadefor/examples/gopath_mode/src/github.com/labstack/echo (from $GOPATH) main.go:5:2: cannot find package "github.com/labstack/echo/middleware" in any of: /Users/kadefor/sdk/go/src/github.com/labstack/echo/middleware (from $GOROOT) /Users/kadefor/examples/gopath_mode/src/github.com/labstack/echo/middleware (from $GOPATH) MacBook$ # 现在就需要下载依赖的包 MacBook$ # 方法一般有: MacBook$ # 1. go get github.com/labstack/echo 它还会同时下载相应的依赖包, 简单 MacBook$ # 但是, 某些包可能因为网络原因访问不了 - -! 可以挂代理 MacBook$ # 2. 想办法把包下载回来解压到`GOPATH/src`目录里, 并保留包目录的结构 MacBook$ # 比如git clone或者去github上点鼠标下载并解压到`GOPATH/src`目录里 MacBook$ # 3. 使用第三方的包管理工具, 比如dep, govendor等 MacBook$ # 第三方包管理工具一般是使用vendor特性, 并支持维护包的版本 MacBook$ # 这样的工具在gopath模式里使用比较多, 因为, go get的方法不支持包版本! MacBook$ # 现在我挂代理使用go get MacBook$ export https_proxy=http://127.0.0.1:7890 http_proxy=http://127.0.0.1:7890 all_proxy=socks5://127.0.0.1:7890 MacBook$ tree `go env GOPATH`/src -L 3 /Users/kadefor/examples/gopath_mode/src └── github.com └── myrepo └── helloworld 3 directories, 0 files MacBook$ go get -v -u -d github.com/labstack/echo github.com/labstack/echo (download) MacBook$ tree `go env GOPATH`/src -L 3 /Users/kadefor/examples/gopath_mode/src ├── github.com │ ├── labstack │ │ ├── echo │ │ └── gommon │ ├── mattn │ │ ├── go-colorable │ │ └── go-isatty │ ├── myrepo │ │ └── helloworld │ └── valyala │ └── fasttemplate └── golang.org └── x ├── crypto ├── net ├── sys └── text 17 directories, 0 files MacBook$ # 其他多出来的就是labstack/echo的依赖包 MacBook$ # 现在运行: MacBook$ go run . ../../labstack/echo/middleware/jwt.go:9:2: cannot find package "github.com/dgrijalva/jwt-go" in any of: /Users/kadefor/sdk/go/src/github.com/dgrijalva/jwt-go (from $GOROOT) /Users/kadefor/examples/gopath_mode/src/github.com/dgrijalva/jwt-go (from $GOPATH) ../../labstack/echo/middleware/rate_limiter.go:9:2: cannot find package "golang.org/x/time/rate" in any of: /Users/kadefor/sdk/go/src/golang.org/x/time/rate (from $GOROOT) /Users/kadefor/examples/gopath_mode/src/golang.org/x/time/rate (from $GOPATH) MacBook$ # 晕! 还有一个包没下载! MacBook$ MacBook$ grep echo/middleware main.go "github.com/labstack/echo/middleware" MacBook$ go get -v -d github.com/labstack/echo/middleware github.com/dgrijalva/jwt-go (download) get "golang.org/x/time/rate": found meta tag vcs.metaImport{ Prefix:"golang.org/x/time", VCS:"git", RepoRoot:"https://go.googlesource.com/time"} at //golang.org/x/time/rate?go-get=1 get "golang.org/x/time/rate": verifying non-authoritative meta tag golang.org/x/time (download) MacBook$ # 再来: MacBook$ go run . ⇨ http server started on [::]:1323 ^Csignal: interrupt MacBook$ # 项目已经运行起来了, 下面再演示一下自己项目下的其他包的使用 MacBook$ pwd /Users/kadefor/examples/gopath_mode/src/github.com/myrepo/helloworld MacBook$ ls main.go MacBook$ mkdir abc MacBook$ pwd /Users/kadefor/examples/gopath_mode/src/github.com/myrepo/helloworld MacBook$ cd abc MacBook$ pwd /Users/kadefor/examples/gopath_mode/src/github.com/myrepo/helloworld/abc MacBook$ # 怎么用这个包呢? 导入路径是什么? MacBook$ # 相对于src的相对路径就是abc这个包的导入路径: MacBook$ # github.com/myrepo/helloworld/abc MacBook$ cat abc.go package abc import "fmt" func Print() { fmt.Println("GOPATH mode: Hello, ABC") } MacBook$ cd .. MacBook$ cat main.go package main import ( "github.com/myrepo/helloworld/abc" "github.com/labstack/echo" "github.com/labstack/echo/middleware" "net/http" ) func main() { abc.Print() e := echo.New() e.Use(middleware.Logger()) e.Use(middleware.Recover()) e.GET("/", hello) e.Logger.Fatal(e.Start(":1323")) } func hello(c echo.Context) error { return c.String(http.StatusOK, "Hello, World!") } MacBook$ cd .. MacBook$ go run main.go GOPATH mode: Hello, ABC ⇨ http server started on [::]:1323 ^Csignal: interrupt MacBook$ # 接下来演示一下vendor MacBook$ pwd /Users/kadefor/examples/gopath_mode/src/github.com/myrepo/helloworld MacBook$ mkdir vendor MacBook$ mkdir -p vendor/github.com/myrepo/helloworld/ MacBook$ cp -a abc vendor/github.com/myrepo/helloworld/ MacBook$ # 改一下vendor下abc那个包, 看看效果 MacBook$ vim vendor/github.com/myrepo/helloworld/abc/abc.go MacBook$ tree . . ├── abc │ └── abc.go ├── main.go └── vendor └── github.com └── myrepo └── helloworld └── abc └── abc.go 6 directories, 3 files MacBook$ cat abc/abc.go package abc import "fmt" func Print() { fmt.Println("GOPATH mode: Hello, ABC") } MacBook$ cat vendor/github.com/myrepo/helloworld/abc/abc.go package abc import "fmt" func Print() { fmt.Println("GOPATH mode vendor: Hello, ABC") } MacBook$ # 运行后输出什么呢? MacBook$ go run . GOPATH mode vendor: Hello, ABC ⇨ http server started on [::]:1323 ^Csignal: interrupt MacBook$ # 我们也可以把所有的依赖包都放到vendor目录下 MacBook$ # 这样的作用是: MacBook$ # 1. 可以把依赖存档, 就算源仓库删除了, 我们的项目同样可以运行 MacBook$ # 2. 保存我们自己修改后的第三方包 MacBook$ # MacBook$ # 但是手动去做太麻烦了, 所以在gopath模式中一般会使用第三方的包管理工具 MacBook$ # 使用主流的第三方包管理工具还有一个好处是: 迁移到gomod模式比较简单! MacBook$gomod 模式(官方建议使用)
1.开启gomod模式, 设置GO111MODULE值为on
MacBook$ # 1. 设置 MacBook$ export GO111MODULE=on MacBook$ # 需要永久配置的话, 需要修改相关的配置文件 MacBook$ # 比如: ~/.bash_profile 或 ~/.bashrc 等 MacBook$ # MacBook$ # 建议使用下面的方法: MacBook$ go env -w GO111MODULE=on MacBook$ MacBook$ # 2. 验证 MacBook$ go env GO111MODULE on MacBook$2.根据需要设置GOPATH, 默认值为~/go, 建议使用默认(这里为了演示设置了其他目录)
MacBook$ # 1. 设置 MacBook$ export GOPATH=/Users/kadefor/examples/gomod_mode MacBook$ # 同上建议: MacBook$ go env -w GOPATH=/Users/kadefor/examples/gomod_mode MacBook$ MacBook$ # 2. 验证 MacBook$ go env GOPATH /Users/kadefor/examples/gomod_mode MacBook$3.设置GOPROXY
MacBook$ # 1. 设置 MacBook$ export GOPROXY=https://goproxy.cn,direct MacBook$ # 同上建议: MacBook$ go env -w GOPROXY=https://goproxy.cn,direct MacBook$ MacBook$ # 有官方的proxy, 但是网站模板网络原因访问不了 MacBook$ # 使用proxy的好处: MacBook$ # 1. 一般公共的proxy都会上CDN, 模块下载速度快 MacBook$ # 2. proxy相当于一个中心化的模块版本镜像, 只要proxy上的缓存不删除 MacBook$ # 就算源仓库删除了, 项目还是可以构建 MacBook$ # MacBook$ # 公司内部如果私有模块比较多, 比较复杂, 可以自建proxy MacBook$ # 由自建的proxy去控制哪些模块是私有的, 哪些是公有的 MacBook$ # 这样对于公司内部的开发来说是透明的, 不需要再关注私有模块 MacBook$ #日常开发(使用labstack/echo这个 web 开发框架为例)
MacBook$ pwd /tmp/helloworld MacBook$ # 使用gomod模式后, 项目就可以随便放在某个目录了, 但是, 项目必须在某个模块内 MacBook$ # 如果是新建的项目, 需要自己创建项目所在模块 MacBook$ go mod init github.com/myrepo/helloworld go: creating new go.mod: module github.com/myrepo/helloworld go: to add module requirements and sums: go mod tidy MacBook$ cat go.mod module github.com/myrepo/helloworld go 1.16 MacBook$ # 这里指定的模块路径为 github.com/myrepo/helloworld MacBook$ # 也可以指定其他的, 比如 abc, xxx/ooo 等等 MacBook$ # 也有限制, 有几个特殊的不行, 哪些? 自己找找 :-) MacBook$ MacBook$ # 这个模块路径一般和源码仓库的路径一致 MacBook$ # 这个模块路径会做为模块目录下包的导入路径的前缀 MacBook$ # 比如, 如果当前模块下有个包abc, 则这个包的导入路径为: MacBook$ # github.com/myrepo/helloworld/abc MacBook$ vim main.go MacBook$ cat main.go package main import ( "github.com/labstack/echo/v4" "github.com/labstack/echo/v4/middleware" "net/http" ) func main() { e := echo.New() e.Use(middleware.Logger()) e.Use(middleware.Recover()) e.GET("/", hello) e.Logger.Fatal(e.Start(":1323")) } func hello(c echo.Context) error { return c.String(http.StatusOK, "Hello, World!") } MacBook$ cat go.mod module github.com/myrepo/helloworld go 1.16 MacBook$ # 使用gomod模式的话, 代码写好了, 可以执行下面命令, 自动下载依赖: MacBook$ # 不需要手动一个一个去下载, 直接执行: MacBook$ go mod tidy go: finding module for package github.com/labstack/echo/v4/middleware go: finding module for package github.com/labstack/echo/v4 go: downloading github.com/labstack/echo/v4 v4.2.0 go: found github.com/labstack/echo/v4 in github.com/labstack/echo/v4 v4.2.0 go: found github.com/labstack/echo/v4/middleware in github.com/labstack/echo/v4 v4.2.0 go: downloading github.com/labstack/gommon v0.3.0 go: downloading golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a go: downloading golang.org/x/net v0.0.0-20200822124328-c89045814202 go: downloading github.com/stretchr/testify v1.4.0 go: downloading github.com/dgrijalva/jwt-go v3.2.0+incompatible go: downloading golang.org/x/time v0.0.0-20201208040808-7e3f01d25324 go: downloading github.com/mattn/go-colorable v0.1.7 go: downloading github.com/mattn/go-isatty v0.0.12 go: downloading gopkg.in/yaml.v2 v2.2.2 go: downloading golang.org/x/sys v0.0.0-20200826173525-f9321e4c35a6 go: downloading golang.org/x/text v0.3.3 MacBook$ # 使用gomod模式的话, 你用的依赖可能有不同的主版本号, 如果是大于等于2, 则在导入路径后面得加上 /v2 /v3 /v4 等 MacBook$ MacBook$ # 我想用labstack/echo的v4版本, 则导入路径为: github.com/labstack/echo/v4 MacBook$ # 现在看看go.mod MacBook$ cat go.mod module github.com/myrepo/helloworld go 1.16 require github.com/labstack/echo/v4 v4.2.0 MacBook$ # 运行 MacBook$ go run . v4.2.0 High performance, minimalist Go web framework ⇨ http server started on [::]:1323 ^Csignal: interrupt MacBook$ # 使用gomod模式还是简单, 只要你的依赖不奇葩 :D MacBook$ MacBook$ # 如果你使用支持自动补全的编辑器或者IDE, 但它不会自动下载依赖(一般都会), 如果模块没有提前下载, 则自动补全无法正常使用 MacBook$ # 或者你需要使用模块特定的版本 MacBook\$ # 那就需要手动下载依赖了: MacBook$ go get -v -d github.com/labstack/echo/v3 go get: module github.com/labstack/echo@upgrade found (v3.3.10+incompatible), but does not contain package github.com/labstack/echo/v3 MacBook$ # 这里需要特别说明一下, 在 gomod 模式出现之前, 有些模块已经有 v2,v3 等版本号了, 但不是模块, 所有就会有类似上面的错误 MacBook$ # 既然不是模块就不存在/v2,/v3这样的尾巴了 MacBook$ go get -v -d github.com/labstack/echo@v3.3.10 go get: added github.com/labstack/echo v3.3.10+incompatible MacBook$ vim main.go # 改一下包导入路径 MacBook$ cat main.go package main import ( "github.com/labstack/echo" "github.com/labstack/echo/middleware" "net/http" ) func main() { e := echo.New() e.Use(middleware.Logger()) e.Use(middleware.Recover()) e.GET("/", hello) e.Logger.Fatal(e.Start(":1323")) } func hello(c echo.Context) error { return c.String(http.StatusOK, "Hello, World!") } MacBook\$ cat go.mod module github.com/myrepo/helloworld go 1.16 require ( github.com/labstack/echo v3.3.10+incompatible // indirect github.com/labstack/echo/v4 v4.2.0 ) MacBook$ go mod tidy MacBook$ # 变化 MacBook\$ cat go.mod module github.com/myrepo/helloworld go 1.16 require ( github.com/dgrijalva/jwt-go v3.2.0+incompatible // indirect github.com/labstack/echo v3.3.10+incompatible github.com/labstack/gommon v0.3.0 // indirect github.com/mattn/go-colorable v0.1.7 // indirect github.com/valyala/fasttemplate v1.2.1 // indirect golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a // indirect golang.org/x/net v0.0.0-20200822124328-c89045814202 // indirect golang.org/x/sys v0.0.0-20200826173525-f9321e4c35a6 // indirect golang.org/x/text v0.3.3 // indirect ) MacBook\$ go run . v3.3.10-dev High performance, minimalist Go web framework ⇨ http server started on [::]:1323 ^Csignal: interrupt04 总结
gomod 模式相对于 gopath 模式来说还是比较新, 所以 gomod 模式下还有很多操作在本文中就没有写了, 如果有人喜欢这种沉浸式的、笔记式的两种、或者视频稿的项目风格, 那后面就再写写 gopath 模式迁移到 gomod 模式的操作, 以及 gomod 模式下模块的常见管理操作, 如果不喜欢这种风格, 那就算了 :D
本文转载自微信公众号「polarisxu」,作者Kade。中代织转载本文请联系polarisxu公众号。码组模式
很赞哦!(65)