美文网首页
go deadlock

go deadlock

作者: SongLiang | 来源:发表于2018-06-01 09:53 被阅读0次

    死锁

    func main() {
        ch := make(chan int)
        <- ch // 阻塞main goroutine, 信道c被锁
    }
    
    fatal error: all goroutines are asleep - deadlock!
    

    何谓死锁? 操作系统有讲过的,所有的线程或进程都在等待资源的释放。如上的程序中, 只有一个goroutine, 所以当你向里面加数据或者存数据的话,都会锁死信道, 并且阻塞当前 goroutine, 也就是所有的goroutine(其实就main线一个)都在等待信道的开放(没人拿走数据信道是不会开放的),也就是死锁咯。

    • 只在单一的goroutine里操作无缓冲信道,一定死锁。比如你只在main函数里操作信道:
    func main() {
        ch := make(chan int)
        ch <- 1 // 1流入信道,堵塞当前线, 没人取走数据信道不会打开
        fmt.Println("This line code wont run") //在此行执行之前Go就会报死锁
    }
    
    • 都在等待流入
    var ch1 chan int = make(chan int)
    var ch2 chan int = make(chan int)
    
    func say(s string) {
        fmt.Println(s)
        ch1 <- <- ch2 // ch1 等待 ch2流出的数据
    }
    
    func main() {
        go say("hello")
        <- ch1  // 堵塞主线
    }
    

    其中主线等ch1中的数据流出,ch1等ch2的数据流出,但是ch2等待数据流入,两个goroutine都在等,也就是死锁。

    • 小结
      其实,总结来看,为什么会死锁?非缓冲信道上如果发生了流入无流出,或者流出无流入,也就导致了死锁。或者这样理解 Go启动的所有goroutine里的非缓冲信道一定要一个线里存数据,一个线里取数据,要成对才行 。
    c, quit := make(chan int), make(chan int)
    
    go func() {
       c <- 1  // c通道的数据没有被其他goroutine读取走,堵塞当前goroutine
       quit <- 0 // quit始终没有办法写入数据
    }()
    
    <- quit // quit 等待数据的写
    

    仔细分析的话,是由于:主线等待quit信道的数据流出,quit等待数据写入,而func被c通道堵塞,所有goroutine都在等,所以死锁。
    简单来看的话,一共两个线,func线中流入c通道的数据并没有在main线中流出,肯定死锁。

    • 所有不成对向信道存取数据的情况都是死锁?
    func main() {
        c := make(chan int)
    
        go func() {
           c <- 1
        }()
    }
    

    程序正常退出了,很简单,并不是我们那个总结不起作用了,还是因为一个让人很囧的原因,main又没等待其它goroutine,自己先跑完了, 所以没有数据流入c信道,一共执行了一个goroutine, 并且没有发生阻塞,所以没有死锁错误。

    • 死锁解决方案
      最简单的,把没取走的数据取走,没放入的数据放入, 因为无缓冲信道不能承载数据,那么就赶紧拿走!
      避免死锁的一种情况:
    c, quit := make(chan int), make(chan int)
    
    go func() {
        c <- 1
        quit <- 0
    }()
    
    <- c // 取走c的数据!
    <-quit
    

    相关文章

      网友评论

          本文标题:go deadlock

          本文链接:https://www.haomeiwen.com/subject/huycsftx.html