Go channel特性
- Go channel的广播机制:
当一个channel被 close时,所有通过select监听这个channel IO事件的goroutine,都会收到相关事件:当前监听的channel已关闭。
func main() {
ch := make(chan int, 2)
for i := 1; i <= 2; i++ {
ch <- i
}
close(ch)
select {
case v1 := <-ch:
fmt.Println("v1: ", v1)
}
select {
case v2 := <-ch:
fmt.Println("v2: ", v2)
}
select {
case v3 := <-ch:
fmt.Println("v3: ", v3)
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 遍历未关闭的channel时会导致死锁阻塞:
如果不close掉channel直接range遍历是会发生死锁的,原因是 当for循环读完channel的10个值之后会继续尝试读取下一个,而由于channel为空又没关闭,会一直阻塞形成死锁。
func main() {
ch := make(chan int, 10)
for i := 1; i <= 10; i++ {
ch <- i
}
fmt.Println("start...")
for v := range ch {
fmt.Println("v: ", v)
}
fmt.Println("end...")
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 被
close后的channel仍然可读,先读出接收到的有效数据,直至channel数据读完后,后续都会读到对应类型的零值,比如:int对应数值0,string对应空字符串 … - 但是如果通过range读取,channel关闭后会直接跳出for循环: 通过
i, ok := <-c中的ok为true还是false,来查看channel的状态,判断值是否有效。