快速掌握Go工作区模式

大家好,我是煎鱼。

创新互联建站是专业的伽师网站建设公司,伽师接单;提供成都网站设计、成都做网站、外贸网站建设,网页设计,网站设计,建网站,PHP网站建设等专业做网站服务;采用PHP框架,可快速的进行伽师网站开发网页制作和功能扩展;专业做搜索引擎喜爱的网站,专业的做网站团队,希望更多企业前来合作!

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

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

背景

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

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

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

// go.mod

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

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

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

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

go work 指令集

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

以下是 go work 的指令集:

go work  [arguments]
  • edit:从工具或脚本中编辑 go.work。
  • init:初始化工作区文件(go.work)。
  • sync:将工作区构建列表同步到模块。
  • use:将模块添加到工作区文件。

快速使用

接下来我们快速应用 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 cdxwcx.com/hello
go: creating new go.mod: module cdxwcx.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 的工作区模式有了一定的了解,其使用场景聚焦在如下:

  • 开发较大的产品,其项目存在着多个互相依赖的模块。可以直接设置成一个工作区。
  • 开发第三方库(类似 SDK 库),需要对上游的模块新增新特性。势必要在本地模块先引用做开发、测试、验证。也可以直接使用工作区。

总结

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

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

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

文章题目:快速掌握Go工作区模式
文章地址:http://www.gawzjz.com/qtweb/news0/206750.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联