如何写 go 代码 (How to Write Go Code 翻译)

简介:

写在前面的话

本文为Go官方网站 How to Write Go Code 这篇文章的翻译,
水平有限, 有些地方不好翻译的地方会用意译的方法, 关于代码示例的路径等方面可能也会有些不一样.

介绍

本文演示如何开发一个简单的 go package, 以及 go tool 的使用方法,
即获取, 编译, 安装 go package 的标准方法和命令.

go tool 需要你安装一定的标准来组织代码. 请仔细阅读本文.
它介绍了用来构建和运行 Go 程序的最简单方法.

介绍本文的视频参照: https://www.youtube.com/watch?v=XCsL89YtqCs (需要FQ!!!)

代码组织

工作区

go tool 是设计用来和公共仓库的开源代码一起工作的.
即使你不需要发布你的代码, go tool 的工作模型也同样适用于你.

Go 代码必须保存在 工作区 中, 工作区 就是一个特定的目录结构, 根目录下有如下3个目录:

  • src 目录: 存放 go 源码文件, 按 package 来组织 (一个 package 一个文件夹)
  • pkg 目录: 存放 package 对象
  • bin 目录: 存放可执行文件

go tool 编译 src 下的文件, 并将编译好的二进制文件分别放入 pkg 或者 bin 文件夹中.
src 目录可以包含多个 VC 仓库(比如 Git 或 Mercurial), 用来管理代码的开发.

下面是一个目录结构的示例:

├── bin                                   # 这里存放 可执行命令
├── pkg                                   # 这里存放 package 对象
│   └── darwin_amd64
│       ├── github.com
│       └── go-files
└── src
    ├── github.com
    │   └── golang
    │       └── example
    │           ├── .git
    │           ├── hello               # 可执行命令的代码
    │           ├── outyet              # 可执行命令的代码
    │           └── stringutil          # package 代码
    └── go-files
        ├── .git
        ├── hello                        # 可执行命令的代码
        └── stringutil                   # package 代码

这里补充说明一下:
src 目录下:

  • github.com/golang/example 是 github 上的代码仓库
  • go-files 是本地的代码仓库

src 目录可以包含多个代码仓库, 可以包含多个命令的源码, 也可以包含多个 package 的源码.
大多数的 Go 程序员会将他们所有的 Go 源码和依赖关系保存在同一个工作区中.

可执行命令 和 库 是分别从不同的 package 的代码编译出来的, 稍后会讨论.

GOPATH 环境变量

GOPATH 环境变量定义了你的 工作区 的位置. 这是你开发 Go 代码时唯一需要设置的环境变量.

开始开发时, 创建 工作区 的文件夹, 并设置对应的 GOPATH 环境变量.
你的 工作区 可以是任意文件夹, 本文中使用的路径是 $HOME/go
注意 不要把 GOPATH 设置为 go 的安装路径.

$ mkdir $HOME/go
$ export GOPATH=$HOME/go

为了方便编译出的命令的执行, 将上面的 bin 目录加入到 PATH:

$ export PATH=$PATH:$GOPATH/bin

Package 路径

标准库中的 package 只要使用短路径即可, 比如 "fmt", "net/http".
对于自己的 package, 必须选一个基本的路径以防止以后和标准库, 或者其他第三方的库产生冲突.

如果你的代码保存在某个代码仓库, 那么就可以使用那个代码仓库的根目录作为你的 package 的基本路径.
比如, 你有个 github 的账户在 github.com/user, 就可以使用 github.com/user 作为你的基本路径.

注意 在能够正确编译代码之前, 你并不需要发布你的代码到远程的代码仓库.
但是如果有一天你会发布代码的话, 好好组织代码结构是个好习惯.
实际上, 你可以使用任意的路径名称, 只要它在 go 标准库和庞大的 go 生态系统中是唯一的.

我们使用 src/go-files 作为基本路径, 然后在工作区中创建文件夹来保存代码

$ mkdir -p $GOPATH/src/go-files

第一个 GO 程序

为了编译和运行一个简单的 GO 程序, 首先要确定 package 路径(这里使用 go-files/hello),
并且在工作区中创建对应 package 文件夹.

$ mkdir $GOPATH/src/go-files/hello

下一步, 在上面文件夹中创建 hello.go 文件, 文件内容如下:

package main

import "fmt"

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

然后, 可以通过 go tool 来编译和安装上面的 hello 程序.

$ go install go-files/hello

你可以在任何路径下运行上述命令, go tool 会根据 GOPATH 环境变量来从工作区中查找 go-files/hello package.

如果在 package 所在文件夹中运行 go install, 也可以省略 package 路径.

$ cd $GOPATH/src/go-files/hello
$ go install

上面的命令编译了 hello 命令, 并产生了此命令的二进制可执行文件.
然后将二进制文件 hello 安装到了 工作区 的 bin 文件夹下(Windows 下是 hello.exe)
在我们的例子中, 就是 $GOPATH/bin/hello, 即 $HOME/go/bin/hello

go tool 只有在出错时才会输出信息, 如果上面的 go 命令没有输出就说明执行成功了.
然后, 就可以在命令行中运行这个命令了.

$ $GOPATH/bin/hello

或者, 如果你将 $GOPATH/bin 加入到 PATH 中了的话, 也可以执行执行 hello 命令.

$ hello

如果你使用了代码版本管理工具, 这时就可以初始化你的仓库, 添加文件, 并 commit 你的第一个改变.
这个步骤是可选的, 写 go 代码并不强制要求使用代码版本管理工具.

$ cd $GOPATH/src/go-files/hello
$ git init
$ git add hello.go
$ git commit -m "initial commit"

发布这个仓库, 使之成为读者的练习仓库.

第一个 GO 库

让我们来写一个库, 并将之用于上面的 hello 程序中.
同样, 首先确定 package 路径 (这里使用 go-files/stringutil), 并创建对应的文件夹.

$ mkdir $GOPATH/src/go-files/stringutil

接着, 创建文件 reverse.go, 内容如下:

// Package stringutil contains utility functions for working with strings.
 package stringutil

 // Reverse returns its argument string reversed rune-wise left to right.
 func Reverse(s string) string {
     r := []rune(s)
     for i, j := 0, len(r)-1; i < len(r) / 2; i, j = i+1, j-1 {
         r[i], r[j] = r[j], r[i]
     }
     return string(r)
 }

用 go build 来编译此 package

$ go build go-files/stringutil

或者在 package 的目录下, 直接运行 go build

$ cd $GOPATH/src/go-files/stringutil
$ go build

上面的命令不会产生输出文件, 为了生成输出文件, 必须使用 go install 命令, 它会在 pkg 文件夹下生成 package 对象.
stringutil package 编译成功之后, 修改之前的 hello.go 文件:

package main

import (
 "fmt"

 "go-files/stringutil"
)

func main() {
     fmt.Printf(stringutil.Reverse("!oG ,olleH"))
}

无论用 go tool 安装 package 对象还是 二进制文件, 它都会安装所有的依赖关系.
所以当你安装 hello 程序时,

$ go install go-files/hello

stringutil package 也会被自动安装.

运行新的 hello 程序, 可以看到如下输出:

$ hello
Hello, Go!

经过上面的步骤, 你的 工作区应该像下面这样:

bin/
    hello                 # command executable
pkg/
    darwin_amd64/          # this will reflect your OS and architecture
        go-files/
            stringutil.a  # package object
src/
    go-files/
        hello/
            hello.go      # command source
        stringutil/
            reverse.go    # package source

注意 go install 将 stringutil.a 放进了 pkg/darwin_amd64 文件夹下 和 代码对应的目录中.
以后, go tool 就可以找到这个 package, 从而判断是否需要重新编译.
darwin_amd64 是表示当前使用的系统, 它的目的是为了区分交叉编译出的其他平台的 package.

Go 编译出的二进制文件都是静态链接的, 所以上面的 bin/hello 在执行时并不需要 darwin_amd64/go-files/stringutil.a 文件.

Package name

go 代码的第一行必须是:

package name

这里的 name 作为 package 的默认名称, 让其他 package import 的时候用.(同个 package 中的所有文件必须使用相同的 name)
Go 的习惯是: package name 是 import path 中最后一部分.
也就是说, 如果一个 package 被引用时写成 "crypto/rot13", 那么这个 package 的 name 就是 rot13

编译为可执行文件的代码的 package name 必须是 main
一个二进制文件所关联的多个 package 的 name 不一定要唯一, 只要 pakage 的 import path 是唯一的就行.
也就是上面的 crypto/rot13 必须唯一, 但是可以有 another-crypto/rot13.

Go 的命名规则可以参考: http://golang.org/doc/effective_go.html#names

测试

Go 中包含一个轻量级的测试框架, 由 go test 命令和 testing package 组成.

测试文件的名称以 _test.go 结尾, 其中包含格式如 func TestXXXX(t *testing.T) 的函数.
测试框架会执行每个这样的函数, 如果函数中调用了 t.Error 或者 t.Fail, 就认为测试失败.

给上面的 package stringutil 增加测试文件, 路径: $GOPATH/src/go-files/stringutil/reverse_test.go, 内容如下:

package stringutil

import "testing"

func TestReverse(t *testing.T) {
  cases := []struct {
      in, want string
  }{
      {"Hello, world", "dlrow ,olleH"},
      {"Hello, 世界", "界世 ,olleH"},
      {"", ""},
  }
  for _, c := range cases {
      got := Reverse(c.in)
      if got != c.want {
          t.Errorf("Reverse(%q) == %q, want %q", c.in, got, c.want)
      }
  }
}

执行测试的方法如下:

$ go test go-files/stringutil

或者进入到 package stringutil 的目录中后, 直接运行:

$ go test

通过 go help test 或者 http://golang.org/pkg/testing/ 来进一步 GO 的测试框架.

远程 package

Go 的 import path 可以描述如何从版本管理系统(Git 或者 Mercurial) 中获取 package 的源码.
go tool 可以利用这个特性来自动获取远程仓库的代码.
比如, 下面的例子中使用的代码同时也保存在 github 上(http://github.com/golang/example).

如果你在代码中 import 了上面这个远程的 package, 那么 go get 命令会自动 获取, 编译, 安装这个 package.

$ go get github.com/golang/example/hello
$ hello
Hello, Go examples!

如果本地没有指定 import 的 package, go get 命令会把这个 package 下载到 GOPATH 中定义的第一个工作区中.
(如果远程 package 不存在的话, go get 相当于 go install)

上面的 go get 命令执行之后, 文件夹结构大致如下:

├── bin
│   └── hello
├── pkg
│   └── darwin_amd64
│       ├── github.com
│       │   └── golang
│       │       └── example
│       │           └── stringutil.a
│       └── go-files
└── src
    ├── github.com
    │   └── golang
    │       └── example
    │           ├── .git
    │           ├── hello
    │           │   └── hello.go
    │           └── stringutil
    │               ├── reverse.go
    │               └── reverse_test.go
    └── go-files
        ├── hello
        │   └── hello.go
        └── stringutil
            ├── reverse.go
            └── reverse_test.go

github.com 上的 hello 程序依赖同一个仓库中的 package stringutil,
即 github.com 上的 hello.go 中引用了 github.com 上的 package stringutil, 所以, go get 命令也下载, 编译, 安装了 stringutil 模块.

import (
  "fmt"

  "github.com/golang/example/stringutil"
)

这个特性可以让你的 go package 很容易的被别人使用.
Go Wiki 和 godoc.org 上列出了很多第三方 Go 工程.
关于使用 go tool 来使用远程仓库的更多信息, 请参考: go help importpath

下一步

  • 订阅 golang-announce 邮件列表来了解最新的 Go release 信息
  • 将 Effective Go 作为参考资料来编写整洁, 地道的 Go 代码
  • 通过 A Tour of Go 来完成一次 go 的旅行
  • 访问 documentation page 来了解一系列关于Go语言的有深度的文章, 以及 Go 库和工具.

获取帮助

  • 寻求实时帮助, 可以使用 FreeNode 的IRC server #go-nuts
  • Go 语言官方邮件列表 Go Nuts
  • 汇报 Go 语言的 bug 请使用 Go issue tracker
标签:  golang



本文转自wang_yb博客园博客,原文链接:http://www.cnblogs.com/wang_yb/p/4333331.html,如需转载请自行联系原作者

目录
相关文章
|
8月前
|
Go 索引
掌握Go语言:Go语言范围,优雅遍历数据结构,简化代码操作实战解析(24)
掌握Go语言:Go语言范围,优雅遍历数据结构,简化代码操作实战解析(24)
|
5月前
|
Cloud Native Go 开发工具
不改一行代码轻松玩转 Go 应用微服务治理
为了更好的进行 Go 应用微服务治理,提高研发效率和系统稳定性,本文将介绍 MSE 微服务治理方案,无需修改业务代码,实现治理能力。
19882 22
|
7月前
|
算法 程序员 编译器
美丽的代码:规范go应用代码注释
【6月更文挑战第30天】本文介绍注释应与代码同步,避免误导,且关键点解释。使用LLVM构建编译器示例展示Go语言规范。注释虽有局限,但在解释复杂逻辑、业务规则时仍有其价值。程序员需平衡注释与代码的关系,创造更优的代码。
1105 0
美丽的代码:规范go应用代码注释
|
2月前
|
安全 Go 开发者
代码之美:Go语言并发编程的优雅实现与案例分析
【10月更文挑战第28天】Go语言自2009年发布以来,凭借简洁的语法、高效的性能和原生的并发支持,赢得了众多开发者的青睐。本文通过两个案例,分别展示了如何使用goroutine和channel实现并发下载网页和构建并发Web服务器,深入探讨了Go语言并发编程的优雅实现。
43 2
|
2月前
|
SQL 监控 算法
为Go应用无侵入地添加任意代码
这篇文章旨在提供技术深度和实践指南,帮助开发者理解并应用这项创新技术来提高Golang应用的监控与服务治理能力。在接下来的部分,我们将通过一些实际案例,进一步展示如何在不同场景中应用这项技术,提供更多实践启示。
|
3月前
|
JSON 搜索推荐 Go
ZincSearch搜索引擎中文文档及在Go语言中代码实现
ZincSearch官网及开发文档均为英文,对非英语用户不够友好。GoFly全栈开发社区将官方文档翻译成中文,并增加实战经验和代码,便于新手使用。本文档涵盖ZincSearch在Go语言中的实现,包括封装工具库、操作接口、统一组件调用及业务代码示例。官方文档https://zincsearch-docs.zinc.dev;中文文档https://doc.goflys.cn/docview?id=41。
145 0
|
5月前
|
缓存 NoSQL 数据库
go-zero微服务实战系列(五、缓存代码怎么写)
go-zero微服务实战系列(五、缓存代码怎么写)
|
5月前
|
程序员 测试技术 Go
用 Go 编写简洁代码的最佳实践
用 Go 编写简洁代码的最佳实践
|
5月前
|
缓存 测试技术 Go
使用Singleflight优化Go代码
使用Singleflight优化Go代码