在一台机器上安装并使用多版本的 Go ?

2,749 阅读4分钟

1. 需求背景

实际开发中会接触到不同的开源项目,而这些项目有可能是不同团队开发的,使用的 go 版本都是不一样的。

以我为例,最近在研究 KubeVirt (一个能在 K8S 下管理虚拟机的开源项目),为了实现自己的需求,需要对 KubeVirt 进行二次开发。

通过 go.mod 得知 KubeVirt 是使用 go 1.13 开发的,而同时为了能在 KubeVirt 下的虚拟机能使用 LVM 这种存储方式,我们又引入了阿里研发的 lvm-csi 插件,同样的,这个插件已经实现的功能距离我们的预期还有点差距,因此同样要进行二次开发。查看 go.mod 发现,lvm-csi 是在更老的版本下开发的,也就是 go 1.12 。

而同时我的机器上很早之前就安装过 go 1.14 的版本,也一直使用它来开发。

问题就来了,如何让才能在我的机器上能同时安装这么多个版本的 golang ,而且互不冲突呢?

2. 安装多版本的 Go

要想不同版本的 go 互不冲突,只要将他们安装在不同的目录下就可以了。

首先下载两个版本的安装包

$ wget https://dl.google.com/go/go1.12.linux-amd64.tar.gz
$ wget https://dl.google.com/go/go1.13.linux-amd64.tar.gz

然后分别解压至不同目录

# 解压go 1.12
sudo tar -C /tmp/ -xzf go1.12.linux-amd64.tar.gz
mv /tmp/go /usr/local/go12 

# 解压go 1.13
sudo tar -C /tmp/  -xzf go1.13.linux-amd64.tar.gz
mv /tmp/go /usr/local/go13

3. 简单粗暴的方案

在同一台机器中安装多个版本的 Golang 需要注意

  • 区分不同版本的 go 入口

  • 切换使用不同的环境变量

区分不同版本 go 入口

当你在环境中有多个 go ,那么当你使用 go build 等命令时,系统如何知道你想要使用的是哪个版本呢?

系统并不知道,因此你需要为不同版本的 go 起不同的名字

  • go 1.12 的 go 入口就改成 /usr/local/go12/bin/go12
  • go 1.13 的 go 入口就改成 /usr/local/go13/bin/go13
  • 以此列推

如此一来,就不会混淆了

$ go12 version
go version go1.12 linux/amd64
$ go13 version
go version go1.13 linux/amd64

切换使用不同的环境变量

使用了不同的go入口,对应的环境变量也要变量,比如 GOROOT,它可不会随着你使用哪个版本的 go 就自动切换,看下面他们输出的都是一个值。

$ go12 env GOROOT
/usr/local/go
$ go13 env GOROOT
/usr/local/go

因此,这个切换不同环境变量得自己来做

另外要注意的是

  1. go 1.12 没有 go env -w 中的 -w 参数
  2. go 1.13 中虽然有 -w 但不能覆盖OS级别的环境变量,修改前需先 unset

知道了这几点之后,便可以自行实现切换的方案。

我想的方案是可以在 /usr/local/go13/bin/ 写一个修改环境变量的脚本

$ cat << EOF >/usr/local/go12/bin/goto12
unset GOROOT
go env -w GOROOT="/usr/local/go12/"
EOF

$ chmod +x /usr/local/go12/bin/goto12

下次要使用 go 1.12 的时候,直接使用下面命令就可以修改环境变量

$ source goto12

go 1.13 也是同理,不敢赘述。

实现的效果如下

$ go env GOROOT
/usr/local/go

$ source goto12
$ go12 env GOROOT
/usr/local/go12

$source goto13
$ go13 env GOROOT
/usr/local/go13

4. 比上面更优的方案

到这里,你应该已经感受到了,每次切换都要手动 source 一下,还是有点麻烦。

其实我还有一个更好的方法,这种方案,可以不用去改 go 入口的名字,同样连 source 都不用。

只要执行下面两条件命令即可(每个版本两条,两个版本就是四条)

$ cat << EOF >/usr/local/go12/bin/go12
unset GOROOT
go env -w GOROOT="/usr/local/go12/"
/usr/local/go12/bin/go $@
EOF

$ chmod + /usr/local/go12/bin/go12

$ cat << EOF >/usr/local/go13/bin/go13
unset GOROOT
go env -w GOROOT="/usr/local/go13/"
/usr/local/go13/bin/go $@
EOF

$ chmod + /usr/local/go13/bin/go13

如果一来,当你执行 go12 就会自动修改 go12 的环境变量,执行 go13 就会自动修改 go13 的环境变量,并且不会影响默认的 go 的行为。

当然以上我都只改了 GOROOT 这一环境变量,若在不同的 go 版本还有其他不一样的地方,可以继续在 go12 或者go13 中添加对应的 shell 代码。

絮叨一下

我在掘金上写过很多的 Python 相关文章,其中包括 Python 实用工具,Python 高效技巧,PyCharm 使用技巧,很高兴得到了很多知乎朋友的认可和支持。

在他们的鼓励之下,我将过往文章分门别类整理成三本 PDF 电子书

PyCharm 中文指南

《PyCharm 中文指南》使用 300 多张 GIF 动态图的形式,详细讲解了最贴合实际开发的 105个 PyCharm 高效使用技巧,内容通俗易懂,适合所有 Python 开发者。

在线体验地址:pycharm.iswbm.com

Python 黑魔法指南

《Python黑魔法指南》目前迎来了 v3.0 的版本,囊集了 100 多个开发小技巧,非常适合在闲时进行碎片阅读。

在线体验地址:magic.iswbm.com

Python 中文指南

学 Python 最好的学习资料永远是 Python 官方文档,可惜现在的官方文档大都是英文,虽然有中文的翻译版了,但是进度实在堪忧。为了照顾英文不好的同学,我自己写了一份 面向零基础的朋友 的在线 Python 文档 -- 《Python中文指南》

在线体验地址:python.iswbm.com

**有帮助的话,记得帮我 点个赞哟~