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
因此,这个切换不同环境变量得自己来做
另外要注意的是:
- go 1.12 没有 go env -w 中的 -w 参数
- 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 代码。