决定开个新坑。java
这个系列首先是关于Go语言实践的。在项目中实际使用Go语言也有段时间了,一个体会就是不管是官方文档、图书仍是网络资料,关于Go语言惯用法(idiom)的介绍都比较少,基本只能靠看标准库源代码本身琢磨,因此我特别想在这方面有一些收集和总结。golang
而后这个系列也是关于设计模式的。虽然Go语言不是一门面向对象编程语言,可是不少面向对象设计模式所要解决的问题是在程序设计中客观存在的。无论用什么语言,老是要面对和解决这些问题的,只是解决的思路和途径会有所不一样。因此我想就以经典的设计模式做为切入点来展开这个系列,毕竟你们对设计模式都很熟悉了,能够避免无中生有想出一些蹩脚的应用场景。编程
本系列的具体主题会比较灵活,计划主要包括这些方面的话题:设计模式
首先要指出的是,绝大多数状况下Go程序是不须要用迭代器的。由于内置的slice和map两种容器均可以经过range进行遍历,而且这两种容器在性能方面作了足够的优化。只要没有特殊的需求,一般是直接用这两种容器解决问题。即便不得不写了一个自定义容器,咱们几乎老是能够实现一个函数,把全部元素(的引用)拷贝到一个slice以后返回,这样调用者又能够直接用range进行遍历了。网络
固然某些特殊场合迭代器仍是有用武之地。好比迭代器的Next()是个耗时操做,不能一口气拷贝全部元素;再好比某些条件下须要中断遍历。闭包
经典实现彻底采用面向对象的思路。为了简化问题,下面的例子中容器就是简单的[]int
,咱们在main
函数中使用迭代器进行遍历操做并打印取到的值,迭代器的接口设计参考java。编程语言
package main import "fmt" type Ints []int func (i Ints) Iterator() *Iterator { return &Iterator{ data: i, index: 0, } } type Iterator struct { data Ints index int } func (i *Iterator) HasNext() bool { return i.index < len(i.data) } func (i *Iterator) Next() (v int) { v = i.data[i.index] i.index++ return v } func main() { ints := Ints{1, 2, 3} for it := ints.Iterator(); it.HasNext(); { fmt.Println(it.Next()) } }
Go语言支持first class functions、高阶函数、闭包、多返回值函数。用上这些特性能够换种方式实现迭代器。函数
初看之下闭包实现与经典实现彻底不一样,其实从本质上来看,两者区别不大。经典实现中把迭代器须要的数据存在struct中,HasNext()
Next()
两个函数定义为Iterator
的方法从而和数据绑定了起来;闭包实现中迭代器是一个匿名函数,它所须要的数据i Ints
和index
以闭包up value的形式绑定了起来,匿名函数返回的两个值正好对应经典实现中的Next()
和HasNext()
。性能
package main import "fmt" type Ints []int func (i Ints) Iterator() func() (int, bool) { index := 0 return func() (val int, ok bool) { if index >= len(i) { return } val, ok = i[index], true index++ return } } func main() { ints := Ints{1, 2, 3} it := ints.Iterator() for { val, ok := it() if !ok { break } fmt.Println(val) } }
这份实现是最go way的,使用了一个channel在新的goroutine中将容器内的元素依次输出。优势是channel是能够用range接收的,因此调用方代码很简洁;缺点是goroutine上下文切换会有开销,这份实现无疑是最低效的,另外调用方必须接收完全部数据,若是只接收一半就中断掉发送方将永远阻塞。优化
依稀记得在邮件列表里看到说标准库里有这个用法的例子,刚才去翻了下没找到原帖了:-)
顺便说一下,“在函数中建立一个channel返回,同时建立一个goroutine往channel中塞数据”这是一个重要的惯用法(Channel Factory pattern,见the way to go 18.8节),能够用来作序列发生器、fan-out、fan-in等。
package main import "fmt" type Ints []int func (i Ints) Iterator() <-chan int { c := make(chan int) go func() { for _, v := range i { c <- v } close(c) }() return c } func main() { ints := Ints{1, 2, 3} for v := range ints.Iterator() { fmt.Println(v) } }
这份迭代器实现是最简洁的,代码也很直白,无须多言。若是想加上中断迭代的功能,能够将func(int)
改成func(int)bool
,Do中根据返回值决定是否退出迭代。
标准库中的container/ring
中有Do()用法的例子。
package main import "fmt" type Ints []int func (i Ints) Do(fn func(int)) { for _, v := range i { fn(v) } } func main() { ints := Ints{1, 2, 3} ints.Do(func(v int) { fmt.Println(v) }) }