Go channel系列:html
双层通道的解释见Go的双层通道并发
如下是一个双层通道的使用示例。注意下面的示例中使用了"信号通道"(Signal channel),但这里的信号通道是多余的,仅仅只是为了介绍。函数
信号通道不用来传递数据,而是用来传递消息,用来产生可读、可写的事件,以便让select选中某个分支。产生消息事件的方式有多种,好比直接关闭通道、发送false/true布尔值等等。code
package main import ( "fmt" "time" ) func main() { // 定义双层通道cc cc := make(chan chan int) times := 5 for i := 1; i < times+1; i++ { // 定义信号通道f f := make(chan bool) // 每次循环都在双层通道cc中生成内层通道c // 并经过信号通道f来终止f1() go f1(cc, f) // 从双层通道cc中取出内层通道ch // 并向ch通道发送数据 ch := <-cc ch <- i // 从ch中取出数据 for sum := range ch { fmt.Printf("Sum(%d)=%d\n", i, sum) } // 每一个循环睡眠一秒钟 time.Sleep(time.Second) // 每次循环都关闭信号通道f close(f) } } // 双层通道cc用来生成内层通道c // 并使用信号通道f来终止函数f1() func f1(cc chan chan int, f chan bool) { c := make(chan int) cc <- c defer close(c) sum := 0 select { // 从内层通道中取出数据,计算和,而后发回内层通道 case x := <-c: for i := 0; i <= x; i++ { sum = sum + i } // goroutine将阻塞在此,直到数据被读走 c <- sum // 信号通道f可读时,结束f1()的运行 // 但由于select没有在for中,该case分支用不上 case <-f: return } }
上面的示例中,函数f1()两个参数,一个是双层通道cc,一个是信号通道f。f1()中首先生成了一个通道c,并发送给了双层通道cc,使得main()中能够从cc中取得这个内层通道c,并向其发送数据。htm
回到f1()中,select最初会被阻塞,由于内层通道c和信号通道f都没有数据可读。因为main()能够取得内层通道c,并向其发送数据,使得f1()中的select第一个case分支被选中,该分支会计算发送的整数以前的总和,并将计算结果从新发送给内层通道c,让main()能够取得这个计算结果。blog
上面的示例中有几个细节须要注意:事件
因此,当在select中有发送操做的时候,极可能会出现死锁现象。这时,要么为select加上default,要么为select加上超时时间,要么select不要放在for循环中。get