阅读 178

Go Select语句和deadlock死锁

在Go语言中,select语句就像switch语句,但是在select语句中,case语句引用通信,即通道上的发送或接收操作。

语法:

select{
    case SendOrReceive1: // 语句    case SendOrReceive2: // 语句    case SendOrReceive3: // 语句    .......
    default: // Statement }

在本文中,我们将学习如何使用默认情况来避免死锁。但是首先,我们了解什么是deadlock?

死锁:当您试图从通道读取或写入数据但通道没有值时。因此,它阻塞goroutine的当前执行,并将控制传递给其他goroutine,但是如果没有其他goroutine可用或其他goroutine睡眠,由于这种情况,程序将崩溃。这种现象称为死锁。如下面的实例所示:

示例

package mainfunc main() {

    //创建通道    //出现死锁是因为没有goroutine在写    //因此,select语句被永远阻塞    c := make(chan int)
    select {
    case <-c:
    }
}

输出:

fatal error: all goroutines are asleep - deadlock!goroutine 1 [chan receive]:main.main()

为了避免这种情况,我们在select语句中使用默认case。换句话说,当程序中出现死锁时,将执行select语句的默认情况以避免死锁。如下例所示,我们在select语句中使用默认情况以避免死锁。

示例

package main 
  import "fmt"  func main() { 
  
    //创建通道    c := make(chan int) 
    select { 
    case <-c: 
    default: 
        fmt.Println("!.. Default case..!") 
    } 
}

输出:

!.. Default case..!

当select语句只有nil通道时,也允许使用默认情况。如下面的实例所示,通道c是nil,所以默认情况下执行,如果这里的默认情况是不可用的,那么程序将永远被阻塞,死锁出现。

示例

package mainimport "fmt"func main() {

    //创建通道    var c chan int    select {
    case x1 := <-c:
        fmt.Println("Value: ", x1)
    default:
        fmt.Println("Default case..!")
    }
}

输出:

Default case..!


文章分类
代码人生
版权声明:本站是系统测试站点,无实际运营。本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 XXXXXXo@163.com 举报,一经查实,本站将立刻删除。
相关推荐