gopl-zh.github.com/ch8/ch8-01.md
2020-09-26 16:31:00 +08:00

2.6 KiB
Raw Permalink Blame History

8.1. Goroutines

在Go语言中每一个并发的执行单元叫作一个goroutine。设想这里的一个程序有两个函数一个函数做计算另一个输出结果假设两个函数没有相互之间的调用关系。一个线性的程序会先调用其中的一个函数然后再调用另一个。如果程序中包含多个goroutine对两个函数的调用则可能发生在同一时刻。马上就会看到这样的一个程序。

如果你使用过操作系统或者其它语言提供的线程那么你可以简单地把goroutine类比作一个线程这样你就可以写出一些正确的程序了。goroutine和线程的本质区别会在9.8节中讲。

当一个程序启动时其主函数即在一个单独的goroutine中运行我们叫它main goroutine。新的goroutine会用go语句来创建。在语法上go语句是一个普通的函数或方法调用前加上关键字go。go语句会使其语句中的函数在一个新创建的goroutine中运行。而go语句本身会迅速地完成。

f()    // call f(); wait for it to return
go f() // create a new goroutine that calls f(); don't wait

下面的例子main goroutine将计算菲波那契数列的第45个元素值。由于计算函数使用低效的递归所以会运行相当长时间在此期间我们想让用户看到一个可见的标识来表明程序依然在正常运行所以来做一个动画的小图标

gopl.io/ch8/spinner

func main() {
	go spinner(100 * time.Millisecond)
	const n = 45
	fibN := fib(n) // slow
	fmt.Printf("\rFibonacci(%d) = %d\n", n, fibN)
}

func spinner(delay time.Duration) {
	for {
		for _, r := range `-\|/` {
			fmt.Printf("\r%c", r)
			time.Sleep(delay)
		}
	}
}

func fib(x int) int {
	if x < 2 {
		return x
	}
	return fib(x-1) + fib(x-2)
}

动画显示了几秒之后fib(45)的调用成功地返回,并且打印结果:

Fibonacci(45) = 1134903170

然后主函数返回。主函数返回时所有的goroutine都会被直接打断程序退出。除了从主函数退出或者直接终止程序之外没有其它的编程方法能够让一个goroutine来打断另一个的执行但是之后可以看到一种方式来实现这个目的通过goroutine之间的通信来让一个goroutine请求其它的goroutine并让被请求的goroutine自行结束执行。

留意一下这里的两个独立的单元是如何进行组合的spinning和菲波那契的计算。分别在独立的函数中但两个函数会同时执行。