简洁的 Go 多版本管理机制

一门充满生机的编程语言,一定是不断进化向前的。随着 Go 项目的持续发展,它目前已经发布到了 1.17 大版本,而且每个大版本内还会有不少小版本的迭代。对于 Go 的版本更新,我们该如何做好多版本管理。

多版本管理的重要性

这里简单列举几个我们需要 Go 多版本管理的理由。

  • 稳定性考量:虽然 Go1 一直在良好地遵守向后兼容准则,但通常基于稳定性考虑,我们并不会直接升级到最新版本。

  • 多项目开发:各项目依赖的 Go 版本不一致。

  • 版本兼容:测试代码前后兼容性,或者确保 bug 修复在不同 Go 版本的正确性,对于开源项目,保证版本兼容性非常重要。

  • 学习新特性:例如虽然我们还在使用 Go 1.16 开发,但是并这不能阻碍你尝鲜 Go 1.17 新功能。

如何多版本管理

我们需要有两个先决条件

  • 已经安装好了某版本的 Go

  • 安装好了 git

安装

运行go install golang.org/dl/go<version> 命令将下载特定 Go 版本的包装器。

$ go install golang.org/dl/go<version>@latest

通过包装器,下载特定 Go 版本和它对应的工具链。

$ go<version> download

例如安装1.14.12版本,可以这样执行。

$ go install golang.org/dl/go1.14.12@latest
$ go1.14.12 download
使用

使用包装器 go1.14.12,我们可以基于 Go v1.14.12 进行构建和测试。

$ go1.14.12 mod init hello
go: creating new go.mod: module hello
$ echo 'package main; import "fmt"; func main() { fmt.Println("Hello, World") }' > hello.go
$ go1.14.12 build
$ ./hello
Hello, World

当然,如果你想让 Go v1.14.12 ”喧宾夺主“,成为 go 命令的代言人,可以这样做。

$ go version
go version go1.17 darwin/amd64
$ export GOROOT=$(go1.14.12 env GOROOT)
$ export PATH=${GOROOT}/bin:$PATH
$ go version
go version go1.14.12 darwin/amd64

这个go1.14.12 env GOROOT 路径就是 Go v1.14.12 版本的内容。所以,如果我们想卸载这个版本,直接将该路径文件夹删除即可;想阅读该版本源码,直接查看该路径下的src/内容即可 。

获取最新开发版本

有一个特别的版本标记:gotip,它用于安装最新的开发版本。

$ go install golang.org/dl/gotip@latest
$ gotip download

可以看到,当前拉取到的最新的开发版本是 go1.18-1afa432。

6575df80d52a6d666935f154c85c2a63.png

实现思路

实现多版本下载安装的秘诀就在于 https://go.googlesource.com/dl 这个仓库,https://github.com/golang/dl 是它的镜像库。

查看仓库代码,我们能看到一系列版本目录

89af06249a80afe3413235c078015124.png

随意选择一个版本进入,会发现存在一个 main.go 文件

a25fb29b772b67b5d390edeb684e1b31.png

而 main.go 文件内容如下

1cdc84e9107845a32bb950ae4ee10622.png

我们通过go install golang.org/dl/go1.14.12@latest下载的 go1.14.12 包装器就是这个 main.go 编译而成。

因此,我们后续通过 go1.14.12 包装器下载和运行的逻辑就在于internal/version包中的 Run 方法了。

// Run runs the "go" tool of the provided Go version.
func Run(version string) {
 log.SetFlags(0)
  // 获取 Go 安装目录
 root, err := goroot(version)
 if err != nil {
  log.Fatalf("%s: %v", version, err)
 }
  
  // 执行 go<version> download 命令时逻辑
 if len(os.Args) == 2 && os.Args[1] == "download" {
  if err := install(root, version); err != nil {
   log.Fatalf("%s: download failed: %v", version, err)
  }
  os.Exit(0)
 }
  
  // 判断该版本 Go 安装状态
 if _, err := os.Stat(filepath.Join(root, unpackedOkay)); err != nil {
  log.Fatalf("%s: not downloaded. Run '%s download' to install to %v", version, version, root)
 }
  // 运行该版本 Go
 runGo(root)
}

鉴于篇幅原因, 下载的install和运行的runGo函数逻辑本文就不再展开了,想深入了解的同学可以自行探索。

另外,为了让每个版本都有一个 Go 包装器主程序(避免重复的手工操作),这里使用了一个帮助命令genv:可以快速生成对应版本的包装器代码<version>/main.go。这里的实现见internal/genv/main.go代码。

总结

本文介绍了 Go 官方提供的多版本管理方案,包括使用、安装、卸载等,可以感受到它的简洁与高效。同时我们简单查看了这一套实现代码逻辑。

最后,希望本文内容能够助你用好 Go 多版本管理,欢迎留言讨论。

ec5e64bcaf10b28dac2465f0355640c2.png

机器铃砍菜刀

欢迎添加小菜刀微信

加入Golang分享群学习交流!

感谢你的点赞在看哦~

d525ba577f17228230ef057c6e306606.gif

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值