为了能更高效地使用语言进行编码,Go 语言有自己的哲学和编程习惯。Go 语言的设计者们从编程效率出发设计了这门语言,但又不会丢掉访问底层程序结构的能力。设计者们通过一组最少的关键字、内置的方法和语法,最终平衡了这两方面。Go 语言也提供了完善的标准库。标准库提供了构建实际的基于 Web 和基于网络的程序所需的所有核心库。
让我们通过一个完整的 Go 语言程序,来看看 Go 语言是如何实现这些功能的。这个程序实现的功能很常见,能在很多现在开发的 Go 程序里发现类似的功能。这个程序从不同的数据源拉取数据,将数据内容与一组搜索项做对比,然后将匹配的内容显示在终端窗口。这个程序会读取文本文件,进行网络调用,解码 XML 和 JSON 成为结构化类型数据,并且利用 Go 语言的并发机制保证这些操作的速度。
读者可以下载本章的代码,用自己喜欢的编辑器阅读。代码存放在这个代码=

没必要第一次就读懂本章的所有内容,可以多读两遍。在学习时,虽然很多现代语言的概念可以对应到 Go 语言中,Go 语言还是有一些独特的特性和风格。如果放下已经熟悉的编程语言,用一种全新的眼光来审视 Go 语言,你会更容易理解并接受 Go 语言的特性,发现 Go 语言的优雅。
2.1 程序架构在深入代码之前,让我们看一下程序的架构(如图 2-1 所示),看看如何在所有不同的数据源中搜索数据。

图 2-1 程序架构流程图
这个程序分成多个不同步骤,在多个不同的 goroutine 里运行。我们会根据流程展示代码,从主 goroutine 开始,一直到执行搜索的 goroutine 和跟踪结果的 goroutine,最后回到主 goroutine。

首先来看一下整个项目的结构,如代码清单 2-1 所示。
代码清单 2-1 应用程序的项目结构
cd $GOPATH/src/github.com/goinaction/code/chapter2

  • sample
  • data
    data.json – 包含一组数据源
  • matchers
    rss.go – 搜索rss源的匹配器
  • search
    default.go – 搜索数据用的默认匹配器 feed.go – 用于读取json数据文件 match.go – 用于支持不同匹配器的接口 search.go – 执行搜索的主控制逻辑 main.go – 程序的入口
    这个应用的代码使用了 4 个文件夹,按字母顺序列出。文件夹 data 中有一个 JSON 文档,其内容是程序要拉取和处理的数据源。文件夹 matchers 中包含程序里用于支持搜索不同数据源的代码。目前程序只完成了支持处理 RSS 类型的数据源的匹配器。文件夹 search 中包含使用不同匹配器进行搜索的业务逻辑。最后,父级文件夹 sample 中有个 main.go 文件,这是整个程序的入口。
    现在了解了如何组织程序的代码,可以继续探索并了解程序是如何工作的。让我们从程序的入口开始。
    2.2 main包程序的主入口可以在 main.go 文件里找到,如代码清单 2-2 所示。虽然这个文件只有 21 行代

码,依然有几点需要注意。
代码清单 2-2 main.go
01 package main 02
03 import (
04 “log”
05 “os”
06
07 _ “github.com/goinaction/code/chapter2/sample/matchers”
08 “github.com/goinaction/code/chapter2/sample/search”
09 )
10
11 // init在main之前调用 12 func init() {
13 // 将日志输出到标准输出
14 log.SetOutput(os.Stdout)
15 }
16
17 // main 是整个程序的入口
18 func main() {
19 // 使用特定的项做搜索
20 search.Run(“president”)
21 }
每个可执行的 Go 程序都有两个明显的特征。一个特征是第 18 行声明的名为main的函数。构建程序在构建可执行文件时,需要找到这个已经声明的main函数,把它作为程序的入口。第二个特征是程序的第 01 行的包名main,如代码清单 2-3 所示。

代码清单 2-3 main.go:第 01 行
01 package main
可以看到,main函数保存在名为main的包里。如果main函数不在main包里,构建工

具就不会生成可执行的文件。
Go 语言的每个代码文件都属于一个包,main.go 也不例外。包这个特性对于 Go 语言来说很重要,我们会在第 3 章中接触到更多细节。现在,只要简单了解以下内容:一个包定义一组编译过的代码,包的名字类似命名空间,可以用来间接访问包内声明的标识符。这个特性可以把不同包中定义的同名标识符区别开。现在,把注意力转到 main.go 的第 03 行到第 09 行,如代码清单 2-4 所示,这里声明了所有的导入项。
代码清单 2-4 main.go:第 03 行到第 09 行
03 import (
04 “log”
05 “os”
06
07 _ “github.com/goinaction/code/chapter2/sample/matchers”
08 “github.com/goinaction/code/chapter2/sample/search”
09 )
顾名思义,关键字import就是导入一段代码,让用户可以访问其中的标识符,如类型、函数、常量和接口。在这个例子中,由于第 08 行的导入,main.go 里的代码就可以引用 search 包里的Run函数。程序的第 04 行和第 05 行导入标准库里的log和os包。

所有处于同一个文件夹里的代码文件,必须使用同一个包名。按照惯例,包和文件夹同名。就像之前说的,一个包定义一组编译后的代码,每段代码都描述包的一部分。如果回头去看看代码清单 2-1,可以看看第 08 行的导入是如何指定那个项目里名叫search的文件夹的。
读者可能注意到第 07 行导入matchers包的时候,导入的路径前面有一个下划线,如代码清单 2-5 所示。
代码清单 2-5 main.go:第 07 行
07 _ “github.com/goinaction/code/chapter2/sample/matchers”
这个技术是为了让 Go 语言对包做初始化操作,但是并不使用包里的标识符。为了让程序的可读性更强,Go 编译器不允许声明导入某个包却不使用。下划线让编译器接受这类导入,并且调用对应包内的所有代码文件里定义的init函数。对这个程序来说,这样做的目的是调用 matchers包中的 rss.go 代码文件里的init函数,注册 RSS 匹配器,以便后用。我们后面会展

示具体的工作方式。

Logo

一起探索未来云端世界的核心,云原生技术专区带您领略创新、高效和可扩展的云计算解决方案,引领您在数字化时代的成功之路。

更多推荐