Golang channel 死锁的几种情况以及例子
死锁是指两个或两个以上的协程的执行过程中,由于竞争资源或由于彼此通信而造成的一种阻塞的现象,若无外力作用,他们将无法推进下去,以下是总结出来的几种死锁情况。
1.死锁1:一个通道在一个主go程里同时进行读和写
2.死锁2:go程开启之前使用通道
3.死锁3 :通道1中调用了通道2,通道2中调用通道1
4.死锁4:直接读取空channel的死锁
5.死锁5:超过channel缓存继续写入数据导致死锁
6.向已关闭的channel中写入数据不会导致死锁,但是回出发panic异常
// 死锁1:一个通道在一个主go程里同时进行读和写
func main() {
// 死锁1
ch := make(chan int)
ch <- 100
num := <-ch
fmt.Println("num=", num)
}
// 死锁2:go程开启之前使用通道
func main() {
ch := make(chan int)
ch <- 100 //此处死锁 优于go程之前使用通道
go func() {
num := <-ch
fmt.Println("num=", num)
}()
//ch <- 100 此处不死锁
time.Sleep(time.Second*3)
fmt.Println("finish")
}
死锁3 :通道1中调用了通道2,通道2中调用通道1
// 死锁3
func main() {
ch1 := make(chan int )
ch2 := make(chan int )
go func() {
for {
select {
case num := <-ch1:
fmt.Println("num=", num)
ch2 <- 100
}
}
}()
for {
select {
case num := <-ch2:
fmt.Println("num=", num)
ch1 <- 300
}
}
}
死锁4:读取空channel 死锁
func main() {
// 死锁1
ch := make(chan int)
//close(ch) 向关闭的channel中读取数据 是该数据的类型的零值
num := <-ch
fmt.Println("num=", num)
}
死锁5:超过channel缓存继续写入数据导致死锁
func main() {
ch := make(chan int, 2)
ch <- 1
ch <- 2
ch <- 3
num := <-ch
fmt.Println("num=", num)
}
死锁6:向已关闭的channel中写入数据不会导致死锁,但是回出发panic异常
func main() {
ch := make(chan int, 2)
close(ch)
ch <- 1
num := <-ch
fmt.Println("num=", num)
}