面试题

这是Go Quiz系列中对于channel的第2篇,波及channel被close后的个性,以及在selectchannel一起应用时的注意事项。

这道题目来源于Google的工程师Valentin Deleplace。

package mainimport "fmt"func main() {    data := make(chan int)    shutdown := make(chan int)    close(shutdown)    close(data)    select {    case <-shutdown:        fmt.Print("CLOSED, ")    case data <- 1:        fmt.Print("HAS WRITTEN, ")    default:        fmt.Print("DEFAULT, ")    }}
  • A: 进入default分支,打印"DEFAULT, "
  • B: 进入shutdown分支,打印"CLOSED, "
  • C: 进入data分支,打印"HAS WRITTEN, "
  • D: 程序会panic
  • E: 程序可能panic,也可能打印"CLOSED, "

这道题次要考查以下知识点:

  • channel被敞开后,从channel接收数据和往channel发送数据会有什么后果?
  • select的运行机制是怎么的?

解析

  1. 对于无缓冲区的channel,往channel发送数据和从channel接收数据都会阻塞。
  2. 对于nil channel和有缓冲区的channel,收发数据的机制如下表所示:

    channelnil空的非空非满满了
    往channel发送数据阻塞发送胜利发送胜利阻塞
    从channel接收数据阻塞阻塞接管胜利接管胜利
    敞开channelpanic敞开胜利敞开胜利敞开胜利
  3. channel被敞开后:

    • 往被敞开的channel发送数据会触发panic。
    • 从被敞开的channel接收数据,会先读完channel里的数据。如果数据读完了,持续从channel读数据会拿到channel里存储的元素类型的零值。

      data, ok := <- c 

      对于下面的代码,如果channel c敞开了,持续从c里读数据,当c里还有数据时,data就是对应读到的值,ok的值是true。如果c的数据曾经读完了,那data就是零值,ok的值是false

    • channel被敞开后,如果再次敞开,会引发panic。
  4. select的运行机制如下:

    • 选取一个可执行不阻塞的case分支,如果多个case分支都不阻塞,会随机选一个case分支执行,和case分支在代码里写的程序没关系。
    • 如果所有case分支都阻塞,会进入default分支执行。
    • 如果没有default分支,那select会阻塞,直到有一个case分支不阻塞。

依据以上规定,本文最开始的题目,在运行的时候

  • data和shutdown这2个channel都被敞开了。
  • 对于敞开的channel,从channel里接收数据,拿到的是channel的存储的元素类型的零值,因而case <-shutdown这个case分支不会阻塞。
  • 对于敞开的channel,向其发送数据会引发panic,因而case data <- 1这个case分支不会阻塞,会引发panic。
  • 因而这个select语句执行的时候,2个case分支都不会阻塞,都可能执行到。如果执行的是case <-shutdown这个case分支,会打印"CLOSED, "。如果执行的是case data <- 1这个case分支,会导致程序panic。

因而本题的答案是E

加餐

能够回顾Go quiz系列中对于channel的第一道题目,加深对channel的了解。

题目链接地址:channel面试题和注意事项

开源地址

文章和示例代码开源地址在GitHub: https://github.com/jincheng9/...

公众号:coding进阶

集体网站:https://jincheng9.github.io/

知乎:https://www.zhihu.com/people/...

References

  • https://twitter.com/val_delep...
  • https://github.com/jincheng9/...
  • https://github.com/jincheng9/...
  • https://github.com/jincheng9/...