golang通道中 nil通道 close通道的特性点

本文详细探讨了Golang中通道(channel)的一些关键特性,包括nil通道的关闭会引发panic,接收者不应关闭通道以免导致发送者在发送时panic,多个发送者时应避免或谨慎关闭通道,不再使用的通道可以依赖GC回收,通道与select的配合使用,以及使用单向通道进行双向通信等。同时,文章提到了Dave Cheney的四个通道公理,强调了对nil通道的操作和关闭通道的影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

英文原文

C1. 以下操作会引起panic
  • P1. Closing the nil channel.

实例:

func C1P1() {
    var ch chan int
    close(ch)
}

结果

panic: close of nil channel

goroutine 1 [running]:
panic(0x458300, 0xc82000a170)
        /home/wdy/go/src/runtime/panic.go:464 +0x3e6
main.main()
        /home/wdy/learn/program-learn/golang/learnchannel.go:27 +0x1e
exit status 2
  • P2. Closing a closed channel.

实例:

func C1P2() {
    ch := make(chan int, 0)
    close(ch)
    close(ch)
}

结果:

panic: close of closed channel

goroutine 1 [running]:
panic(0x458300, 0xc82000a170)
        /home/wdy/go/src/runtime/panic.go:464 +0x3e6
main.main()
        /home/wdy/learn/program-learn/golang/learnchannel.go:27 +0x4d
exit status 2
  • P3. Sending on a closed channel.

实例:

func C1P3() {
    ch := make(chan int, 0)
    close(ch)
    ch <- 2
}

结果:

panic: send on closed channel

goroutine 1 [running]:
panic(0x458300, 0xc82000a170)
        /home/wdy/go/src/runtime/panic.go:464 +0x3e6
main.main()
        /home/wdy/learn/program-learn/golang/learnchannel.go:23 +0x6c
exit status 2
C2. 作为接受者的goroutine不要关闭channel,否则 发送者未来往这个channel发送数据时会panic
C3. 如果一个channel有多个发送者,发送者就不要关闭通道或者最后一个发送数据的goroutine来关闭通道。

Last one sender to leave, turns off the lights, which can be controlled by a atomic int

C4. 对于不再使用的通道不必显示关闭。如果没有goroutine引用这个通道,这个通道就会被垃圾回收。

注意如果需要把关闭通道作为一个控制信号告知其他goroutine没有更多数据的情况下,需要显示关闭。讨论

C5. 通道和select搭配最佳。
C6. 如果在两个goroutine中需要一个双向沟通,考虑使用两个单独的单向通道。这样两个通道就可以通过调用close的方法来告知对方通讯终止。
C7. 如果通道没有负责读取的goroutine的时候,那么发送者就可能会永远阻塞在该通道上。
C8. 当设计一个goroutine通过channel来提供服务的时候,在某个时刻这个goroutine不再需要了,要仔细考虑goroutine应该怎么结束。否则这个不再使用的goroutine会服务一个不受监管的channel。
C9. Dave Cheney’s Four Channel Axioms :
  • A1. 对一个 nil channel执行发送操作会一直阻塞。

实例:

func C9A1() {
    fmt.Println("C9A1")
    var ch chan int
    ch <- 2
}

结果:

C9A1
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan send (nil chan)]:
main.C9A1()
        /home/wdy/learn/program-learn/golang/learnchannel.go:27 +0xf9
main.main()
        /home/wdy/learn/program-learn/golang/learnchannel.go:37 +0x14
exit status 2

发送:

func C9A2() {
    fmt.Println("C9A2")
    var ch chan int
    <-ch
}

结果:

C9A2
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [chan receive (nil chan)]:
main.C9A2()
        /home/wdy/learn/program-learn/golang/learnchannel.go:33 +0xec
main.main()
        /home/wdy/learn/program-learn/golang/learnchannel.go:38 +0x14
exit status 2
  • A3. 发送到关闭的channel会引起panic

  • A4. 从关闭的cannel读操作,会立刻返回数据0值。

C10. ‘select’ 从来不会选择阻塞的case,意思就是如果select的多个case都阻塞了,那么当前goroutine就会阻塞在select上。

后记

之前一直在优快云上写文章,后面会逐步转换到简书上,还请大家多多支持。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值