如何从源代码构建 Go 1.5 开发环境

b2f5 10年前

近期,Go Team 连续放出了几个大招来介绍即将在八月问世的 Go 1.5 这个划时代的版本。Rob 和 Andrew 分别在《Go in Go》和《The State of Go》中详细说明了出现在 Go 1.5 中的重要特性和细节变化。在这个版本中最主要的变化是移除了所有 C 代码,不论是 runtime 还是编译器都使用 Go 语言和一小部分的汇编来实现——也就是人们常说的自举。但是这样做也就意味着,Go 在 1.5 和以后的版本中,使用源代码构建 Go
开发环境将面临“鸡生蛋,蛋生鸡”的麻烦(当然了,如果你直接“买鸡蛋”——使用二进制安装包——是没有这个问题的)。

在 Go1.4 及更早的版本中,会使用 GCC 先编译一个使用 C 语言编写的,仅具有基本功能的小编译器作为构建 Go 环境的引导工具。也就是说必须要安装 GCC、make 等 C 语言相关的工具才能从源代码构建 Go 的开发环境。而据 Rob 的讲义和其撰写的《Go 1.5 Bootstrap Plan》中介绍的,Go 1.5 将不再有 C 语言的参与,反而需要使用 Go 1.4 版本的工具链进行编译。那么也就意味着,从源代码开始构建 Go 1.5 需要两个版本并存。几年前,有许多人折戟在 GOROOT/GOPATH 的坑里。现在还需要两个版本的 Go 并存,想想似乎都是个挺麻烦的事情。

对于 Go 来说,大道至简!所以通过这篇文章里我想简单介绍一下如何使用源代码构建 Go 1.5 开发环境。由于“鸡生蛋,蛋生鸡”的缘故,需要从构建 Go 1.4 的开发环境开始讲起。

需要说明的是,以下所有内容都是在 Ubuntu 14.04 中演示操作的,但只要是符合 POSIX 标准的系统,以下操作应该都是一致的。Windows 的用户我强烈建议还是使用二进制包进行安装。不折腾!

准备工作

一个“干净”的系统是必须的,这里的“干净”是指没有设置过 GOROOT/GOPATH/GOBIN 之类的环境变量。如果之前已经配置过 Go 的环境,那只能酌情调整或删除重新设置了。

同时,由于需要编译 Go 1.4,所以必须安装 C 相关的工具:

$ apt-getinstallgcc libc6-dev
</div> </div>

##目录结构

2012 年的时候,我曾经翻译过一篇文章《GO 环境设置》。虽然那个时候 Go 的代码还在使用 hg 进行版本控制,同时 Go 1 也没有正式发布,不过那篇文章中介绍的目录设置方式,我却一直使用至今,其结构如下:

$HOME/golang/
├── 3rdpkg
├── go
└── own
</div> </div>

其中$HOME/golang/3rdpkg,$HOME/golang/go和$HOME/golang/own目录应按照顺序加入环境变量GOPATH中。这样的好处是在使用go get获取 Go 包的时候会直接导入到GOPATH的第一个路径,也就是3rdpkg这个子目录中。这样可以将第三方包,Go 的代码和自己的工作目录区分开来。

不过,由于 Go 1.5 需要两个版本的 Go 并存,那么这个目录结构也就需要做相应的调整。最终如下,稍候我会详细介绍。

环境变量

前面已经提到了 GOPATH 的设置:

GOPATH=$HOME/golang/3rdpkg:$HOME/golang/go:$HOME/golang/own
</div> </div>

由于现在有两个版本的 Go 代码并存,所以我们需要建立一个软链接指向所需要的版本的代码目录,例如:

$HOME/golang/
├── 3rdpkg
├── go -> go1.4/
├── go1.4
└── own
</div> </div>

这样,就 GOROOT 的值就应该设置为:

GOROOT=$HOME/golang/go
</div> </div>

有了这两个环境变量就足够了(交叉编译和环境微调不在本文讨论范围内)。

为了能够方便的使用 go 命令,还需要将$GOROOT/bin/加入 PATH 中:

PATH=$PATH:$GOROOT/bin/
</div> </div>

安装 Go 1.4

使用 git 命令获取 Go 1.4 的完整代码。当前最新的 1.4 版本是 1.4.2,所以:

$cd$HOME/golang/
$ git clone -b go1.4.2 https://github.com/golang/go.git go1.4
</div> </div>

然后让 GOROOT 的软链接目录指向实际保存 Go 1.4 代码的目录:

$ln-s go1.4 go
</div> </div>

这时目录结构为:

$HOME/golang/
├── 3rdpkg
├── go -> go1.4/
├── go1.4
└── own
</div> </div>

环境变量的值为:

GOPATH=$HOME/golang/3rdpkg:$HOME/golang/go:$HOME/golang/own
GOROOT=$HOME/golang/go
PATH=$PATH:$GOROOT/bin
</div> </div>

进入目录$HOME/golang/go/src,运行 all.bash 脚本。

cd$HOME/golang/go/src
./all.bash
</div> </div>

经过一个短暂的编译和一个漫长的测试之后,Go 1.4 应该就部署完成了。

使用go version命令可以看到当前 Go 版本为 1.4.2:

$ go version go1.4.2 linux/amd64
</div> </div>

安装 Go 1.5

由于 Go 1.5 需要基于 Go 1.4 构建,所以 Go 1.5 需要一个独立的目录放置(实际上用同一个目录是可以的,不过需要额外的许多设置,不折腾)。

由于已经克隆了 Go 的代码库,可以直接复制 go1.4 这个目录到目录 go1.5,然后用命令:

$ go checkout -b master
</div> </div>

切换到 Go 1.5 所在的代码分支中。

有一点需要特别说明一下:由于 Go 1.5 预计要到八月才正式发布,所以要到那个时候才会有 go1.5 这个标签出现。因此,当前的 master 分支实际上就是功能冻结的 Go 1.5 的代码分支。

由于已经将 GOROOT 设置为$HOME/golang/go,因此只需要这个软链接重新指向 Go 1.5 代码所在目录即可,而无须修改环境变量。

$cd$HOME/golang
$ unlink go
$ln-s go1.5 go
</div> </div>

为了能够编译 Go 1.5,还需要额外设置一个叫做 GOROOT_BOOTSTRAP 环境变量,指向 Go 1.4 所在的目录。同时为了能够向后兼容,这个变量也使用软链接的方式进行指向:

$ln-s go1.4 go-bootstrap
</div> </div>

这时目录结构为:

$HOME/golang/
├── 3rdpkg
├── go -> go1.5/
├── go-bootstrap -> go1.4/
├── go1.4
├── go1.5
└── own
</div> </div>

环境变量的值为:

GOPATH=$HOME/golang/3rdpkg:$HOME/golang/go:$HOME/golang/own
GOROOT=$HOME/golang/go
PATH=$PATH:$GOROOT/bin
GOROOT_BOOTSTRAP=$HOME/golang/go-bootstrap
</div> </div>

进入目录$HOME/golang/go/src,运行 all.bash 脚本。

$cd$HOME/golang/go/src
$ ./all.bash
</div> </div>

又一个短暂的编译和一个漫长的测试之后,Go 1.5 应该就部署完成了。

使用go version命令可以看到当前 Go 版本为开发编号:

go version devel +6551803 Wed May 27 04:48:29 2015 +0000 linux/amd64
</div> </div>

对于持续跟踪 Go 代码库的开发人员来说,保持这样的一个目录结构可以让以后的升级变得更为轻松。只需要建立新版本的代码目录,并调整软连接的指向,然后编译即可得到新版本的开发环境。

来自:http://mikespook.com/2015/06/%E5%A6%82%E4%BD%95%E4%BB%8E%E6%BA%90%E4%BB%A3%E7%A0%81%E6%9E%84%E5%BB%BA-go-1-5-%E5%BC%80%E5%8F%91%E7%8E%AF%E5%A2%83/