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

请注意,本文正文含有大量链接。如果是转载或者使用某些不支持超链接的阅读器,就请自行脑补吧。

近期,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-get install gcc libc6-dev

##目录结构

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

$HOME/golang/
├── 3rdpkg
├── go
└── own

其中 $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

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

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

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

GOROOT=$HOME/golang/go

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

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

PATH=$PATH:$GOROOT/bin/

安装 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

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

$ ln -s go1.4 go

这时目录结构为:

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

环境变量的值为:

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

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

cd $HOME/golang/go/src
./all.bash

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

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

$ go version go1.4.2 linux/amd64

安装 Go 1.5

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

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

$ go checkout -b master

切换到 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

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

$ ln -s go1.4 go-bootstrap

这时目录结构为:

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

环境变量的值为:

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

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

$ cd $HOME/golang/go/src
$ ./all.bash

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

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

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

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

Join the Conversation

13 Comments

  1. 下载1.4的bin包,解压到$HOME,把解压出来的go目录改名成go1.4,就可以在master分支里跑all.bash编译了。

  2. 既然要自己编译Go1.5,Go1.4这只鸡还是买二进制版本比较方便 🙂

  3. 二进制直接装肯定是方便的,不过像我这样一路从0.x 的 weekly 升级上来的,有时需要环境有一定的延续性。要不 VIM 什么的一大堆都要做调整。

  4. 1.5 要支持动态链接了,我已经看到一大波打包即将到来……

  5. 我按照你的方法,设置了own目录,在own目录下设置,src,pkg,bin.
    idea不认识在$HOME/golang/3rdpkg目录下的第三方go文件,而还是认识在$HOME/golang/own/src下的第三方go文件(它们是自动生成的,我尝试删除它们,不过失败了)
    我的$GOPATH是/home/lily/golang/3rdpkg:/home/lily/golang/go:/home/lily/golang/own
    但是s使用go get的命令 ,电脑的导入文件的目录是 $HOME/golang/own/src

    请问哪里的设置导致了这个问题了呢?感谢!

  6. 0.0已经解决了。
    src,pkg,bin.是在3rdpkg生成。(自己反复看文件,折腾)
    谢谢。

  7. 你好。我的$GOPATH是
    /home/lily/golang/3rdpkg:/home/lily/golang/go:/home/lily/golang/own
    但是idea只导入前面两个路径,而
    /home/lily/golang/own
    这个路径,idea没有导入
    。。。

  8. 我一般是用GVim,所以不太清楚 idea 是怎么使用的。正常来说,这个是系统的环境变量。如果系统能够读取到,应该不会有错。

  9. hello. 在opencart官网下载了你开发的Qiniu CDN Lite(http://www.opencart.com/index.php?route=extension/extension/info&extension_id=23525),有几个问题请教你,方便加下QQ:584236900,谢谢!

  10. 折腾了好久,总是有import cycle package 的问题。。。去官方下个编译好的就搞定了!

Leave a comment

Your email address will not be published. Required fields are marked *