361ad5a5fd6346731f93b557a5619f97.png

问题

已经关闭的的 chan 进行读写,会怎么样?为什么?

怎么答

  • 已经关闭chan 能一直读到东西,但是读到的内容根据通道内关闭前是否有元素而不同。
    • 如果 chan 关闭前,buffer 内有元素还未读 , 会正确读到 chan 内的值,且返回的第二个 bool 值(是否读成功)为 true
    • 如果 chan 关闭前,buffer 内有元素已经被读完chan 内无值,接下来所有接收的值都会非阻塞直接成功,返回 channel 元素的零值,但是第二个 bool 值一直为 false
  • 已经关闭chanpanic

举例

1. 写已经关闭的 chan

cf017a185707cb929d998a57e57471ef.png
  • 注意这个 send on closed channel,待会会提到。

2. 读已经关闭的 chan

4ac7f90fd23b30ef3f4ec7ddcac12b35.png

多问一句

1. 为什么写已经关闭的 chan 就会 panic 呢?

fc30379bb2d2e0c6d552fecc8dfac6cc.png
  • c.closed != 0 则为通道关闭,此时执行写,源码提示直接 panic,输出的内容就是上面提到的 "send on closed channel"

2. 为什么读已关闭的 chan 会一直能读到值?

5571da6debb3bd498a97872433f37e90.png
  • c.closed != 0 && c.qcount == 0 指通道已经关闭,且缓存为空的情况下(已经读完了之前写到通道里的值)
  • 如果接收值的地址 ep 不为空
    • 那接收值将获得是一个该类型的零值
    • typedmemclr根据类型清理相应地址的内存
    • 这就解释了上面代码为什么关闭的 chan 会返回对应类型的零值

文章推荐:

  • 对未初始化的的 chan 进行读写,会怎么样?为什么?
  • golang 面试题:​reflect(反射包)如何获取字段 tag​?为什么 json 包不能导出私有变量的 tag?
  • golang 面试题:json 包变量不加 tag 会怎么样?
  • golang 面试题:怎么避免内存逃逸??
  • golang 面试题:简单聊聊内存逃逸?
  • golang 面试题:字符串转成 byte 数组,会发生内存拷贝吗?
  • golang 面试题:翻转含有中文、数字、英文字母的字符串
  • golang 面试题:拷贝大切片一定比小切片代价大吗?
  • golang 面试题:能说说 uintptr 和 unsafe.Pointer 的区别吗?

如果你想每天学习一个知识点,关注我的【公】【众】【号】【golang 小白成长记】。

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐