有一道这样的面试题目:golang
写代码实现两个 goroutine,其中一个产生随机数并写入到 go channel 中,另一个从 channel 中读取数字并打印到标准输出。最终输出五个随机数。
凭直觉开始撸了如下代码:面试
func getRand(ch chan int) { ch <- rand.Intn(5) } func printRand(ch chan int) { fmt.Println("Rand Number = ", <-ch) } func main() { rand.Seed(63) ch := make(chan int) for i := 1; i <= 5; i++ { go getRand(ch) go printRand(ch) } }
这个是大部分人凭直觉写出来的,稍微有经验的就会发现问题。执行起来会什么都不打印程序就退出了,由于 main 方法没有等 goroutine 退出就结束了。下面改造一下 main 方法。dom
func main() { rand.Seed(63) ch := make(chan int) for i := 1; i <= 5; i++ { go getRand(ch) go printRand(ch) } time.Sleep(3 * time.Second) // 加这一行等待 goroutine close(ch) }
如今至少输出达到要求了,可是还有问题。code
func main() { ch := make(chan int) done := make(chan bool) go func() { for { select { case ch <- rand.Intn(5): // Create and send random number into channel case <-done: // If receive signal on done channel - Return return default: } } }() go func() { for i := 0; i < 5; i++ { fmt.Println("Rand Number = ", <-ch) // Print number received on standard output } done <- true // Send Terminate Signal and return return }() <-done // Exit Main when Terminate Signal received }
上面是一个比较完美的答案,通常推荐使用匿名方法建立 goroutine,经过 Done channel 来实现 goroutine 之间的通知符合 Golang 的思惟模式。 get
思考:it