Skip to content
Name5566 edited this page Jun 12, 2016 · 1 revision

workspace

Golang 的代码必须放置在一个 workspace 中。一个 workspace 是一个目录,此目录中包含几个子目录:

  1. src 目录。包含源文件,源文件被组织为包(一个目录一个包)
  2. pkg 目录。包含包目标文件(package objects)
  3. bin 目录。包含可执行的命令

包源文件(package source)被编译为包目标文件(package object),命令源文件(command source)被编译为可执行命令(command executable)。使用 go 命令进行构建生成的包目标文件位于 pkg 目录中,生成的可执行命令位于 bin 目录中。开发 Golang 需要设置一个环境变量 GOPATH,此环境变量用于指定 workspace 的路径,另外,也可以把 workspace 的 bin 目录加入到 PATH 环境变量中去。

包路径

包位于 src 目录下,只要不发生冲突(例如和标准库冲突),我们可以使用任意的包路径。假如我们在 GitHub 有一个账号 name5566,并且存在一个项目 hello,那么我们可以在 src 目录下构建这样的目录结构:

github.com/name5566/hello

在 import 中使用此包路径来导入相关的包。

第一个可执行命令

现在来编写第一个 Golang 程序 hello。假定已经配置好了 GOPATH,并且包路径使用 github.com/name5566/hello,在此目录下创建文件 hello.go:

package main

import "fmt"

func main() {
    fmt.Printf("Hello, world.\n")
}

执行命令(可在任意处执行):

go install github.com/name5566/hello

这个命令可以编译 hello 命令,生成一个可执行命令 hello 并放置于 bin 目录下。如果 go 命令没有任何输出表示执行成功。

第一个包目标文件

现在开始创建一个包目标文件(也就是库文件),首先建立包路径 github.com/name5566/newmath,在此目录下创建文件 sqrt.go:

package newmath

func Sqrt(x float64) float64 {
    z := 1.0
    for i := 0; i < 1000; i++ {
        z -= (z*z - x) / (2 * z)
    }
    return z
}

我们可以测试编译此包:

go build github.com/name5566/newmath

此命令不会有任何输出文件,如果需要在 pkg 目录下生成包目标文件,使用 go install。

现在来修改一下 hello.go:

package main

import (
    "fmt"
    "github.com/name5566/newmath"
)

func main() {
    fmt.Printf("Hello, world. Sqrt(2) = %v\n", newmath.Sqrt(2))
}

编译安装 hello:

go install github.com/name5566/hello

go 命令能够自己分析依赖关系,这里的 newmath 包会被自动安装到 pkg 目录下。

另外需要注意的是,Golang 使用静态链接(运行 Go 程序无需包目标文件)。

测试

Golang 带有轻量级测试框架,主要包括:

  1. go test 命令
  2. testing 包

创建文件 _test.go 来编写测试,例如为 sqrt.go 创建的测试文件为 sqrt_test.go:

package newmath

import "testing"

func TestSqrt(t *testing.T) {
    const in, out = 4, 2
    if x := Sqrt(in); x != out {
        t.Errorf("Sqrt(%v) = %v, want %v", in, x, out)
    }
}

执行测试使用 go test:

go test github.com/name5566/newmath

测试文件中包含的所有测试函数 func TestXxx(t *testing.T) 都会被执行。测试函数可以通过 Error、Fail 等相关方法告知出现错误。

另外,在测试文件中还可以编写范例代码,例如:

func ExampleHello() {
    fmt.Println("hello")
    // Output: hello
}

这里 Example 函数结尾可以跟上一个以 “Output:” 字符串开始的注释(被叫做输出注释),在测试运行时会将 Example 函数输出注释中字符串和函数标准输出进行比对。需要注意的是,如果没有输出注释,Example 函数是不会被运行的(但是会被编译)。

Example 函数命名有这样的习惯:

func Example() { ... }
func ExampleF() { ... }
func ExampleT() { ... }
func ExampleT_M() { ... }

这里的 F 为函数名,T 为类型名,T_M 为类型 T 上的方法 M。

获取远程仓库的包

使用 go get 能够自动获取并安装远程仓库的包,例如:

go get code.google.com/p/go.example/hello