快速掌握 Go 工作区模式

425次阅读  |  发布于1年以前

在 Go 项目的模块管理中,先是 GOPATH,然后到废弃。再到强推 Go modules,从被社区抗拒到 rsc 硬上弓。现在最新要了解的,就是工作区模式(workspace mode)。这是一个在 Go1.18 引入的重要特性。

之前一直没提过,今天补全这块的知识点。

背景

在 Go1.11 起有了 Go modules 后,看起来 Go 模块管理逐步按序有了约束、规范了起来。但也带来了一些使用上的问题。

现实开发时,当我们需要对多个关联模块进行开发(修改)时,这个事情就麻烦了起来。我见过两种方式。

1、第一种:直接在 go.mod 文件上配置 replace,配置到本地的开发目录。这是最常见的方式。

// go.mod

replace example.com/golang/text => "../eddycjy/golang/text"

这种做法经常会有人不小心提交到 Git 仓库上。还挺折腾人的,一个不小心就为此 debug 了半天,或者发布部署一直卡着过不去。

2、第二种:直接在依赖模块上编码,编码到一定的程度。才上传 GitHub/GitLab。再去发布版本标签再引用。这种用法比较少,只有模块比较简单且对程序比较自信的会这么干。(不推荐)

总的来讲,就是有了 Go modules 后,多模块间的依赖开发还是挺麻烦的。要经常 replace,有时候又会忘了删。

go work 指令集

在大家痛苦了许久后,Go1.18 时终于发布了工作区模式的方式,来优化这个用法和问题。

以下是 go work 的指令集:

go work <command> [arguments]

快速使用

接下来我们快速应用 Go 工作区模式,让大家有个直观的了解。

需要注意,该特性需要确保 Go 版本 >= 1.18。

创建工作区

首先我们创建一个工作区,执行如下命令:

$ mkdir workspace-main && cd workspace-main 
$ go work init

执行完毕后会在该目录下创建一个 go.work 文件,文件内容包含:

go 1.20

仅包含版本信息,因为当前是空白的工作区,只有初始化行为。

创建演示模块

$ mkdir hello-world && cd hello-world
$ go mod init example.com/hello
go: creating new go.mod: module example.com/hello

写入代码 hello.go:

package main

import (
 "fmt"

 "golang.org/x/example/hello/reverse"
)

func main() {
 fmt.Println(reverse.String("Hello, 煎鱼"))
}

如果你这时候直接 go run。可能会出现如下报错:

hello.go:6:5: no required module provides package golang.org/x/example/hello/reverse: go.mod file not found in current directory or any parent directory; see 'go help modules'

看着非常迷惑人,很多同学以为是环境变量 GO111MODULE 没有设置为 on。其实是没有将本模块加入工作区中,导致运行错误。

所以可以看出来,在设计上是先有项目,再有工作区的路径。也是相对符合的。

这时候需要回到工作区目录 workspace-main。执行如下命令:

go work use ./hello-world

go.work 文件内会变成:

$ cat go.work 
go 1.20

use ./hello-world

再运行程序:

$ go run hello-world/hello.go 
鱼煎 ,olleH

一切正常。

创建需修改的模块

这时候我们有了一个实际的诉求,我们希望 golang.org/x/example/hello 改一下这个 SDK 库。

如果是以前的话,我们需要写 replace 来解决。现在的话可以用工作区模式来完成这个诉求。

我们先需要回到工作区根目录 workspace-main 下,拉取这个 SDK 库到工作区中:

git clone https://go.googlesource.com/example

再将其引入项目的工作区中:

go work use ./example/hello

go.work 文件会变成:

go 1.20

use (
 ./example/hello
 ./hello-world
)

这里需要注意,go work 以 go.mod 为单位。如果你直接引入 ./example。是无法对 ./example/hello 的 module 起效果的。

在引入成功后,我们回到 ./example/hello 目录下的 reverse.go 文件,新增一个用于 Demo 的方法:

...
func Hello() string {
 return "煎鱼,你好!"
}

再到 hello 项目中,新增调用:

package main

import (
 "fmt"

 "golang.org/x/example/hello/reverse"
)

func main() {
 fmt.Println(reverse.String("Hello, 煎鱼"))
 fmt.Println(reverse.Hello())
}

输出结果:

鱼煎 ,olleH
煎鱼,你好

一切正常。满足不添加 replace 的要求,也使用了 go.work,不用担心把 replace 不小心提交到 Git 仓库中。

另外 Go 工作区中的项目在进行编译时,也是引用所配置好的工作区内的模块。而不是单单只针对开发阶段的 go run,也可以在产线上去使用,编译成二进制去应用和部署。

场景汇总

我们已经对 Go 的工作区模式有了一定的了解,其使用场景聚焦在如下:

总结

今天我们快速了解了 Go 工作区模式(workspace mode)的背景、使用、场景。这对于解决项目中多模块依赖有着一定的作用,可以不再需要去 go.mod 里 replace,算是给了一个规范化的解决方案。

但在实际应用中,我们会发现工作区模式的便利度,其实不太高。可能依赖模块数量少时,还不如 replace 一把梭来得快。

另外目前阶段的使用宣传还是做得比较弱的,前两天问了一圈,还真有一些同学不知道,也没有用过的。

Copyright© 2013-2020

All Rights Reserved 京ICP备2023019179号-8