对于无缓冲的 channel,用 select 来处理通道超时时间,如果在接收超时后,退出接收,那么该通道的发送会被阻塞吗?
会阻塞吗?实现原理是什么?
测试代码如下:
package main
import (
"fmt"
"time"
)
func main() {
c1 := make(chan string)
go func{
time.Sleep(2 * time.Second)
c1 <- "this is send"
fmt.Println("send ok")
}()
select {
case res := <- c1:
fmt.Println(res)
case <- time.After(1 * time.Second):
fmt.Println("timeout")
}
time.Sleep(2 * time.Second)
fmt.Println("over")
}
有疑问加站长微信联系(非本文作者)

对于无缓冲的channel,发送时会阻塞在发送位置直到被接收后程序继续往下执行,接收时会阻塞在接收位置直到接收到信息然后程序继续往下执行。是否阻塞和时间无关,是为了某种情景下认为的控制。就本例而言,“this is send"写入channel中,此时select模块已经执行完成,消息没有被接收,因此"send ok"直到程序结束也不会被输出。
就本例而言,"send ok" 确实不会输出,但是 <- chan 是如何知道此时 recv 不存在(没有就绪)?
我觉得阻塞主要是对接收方来说的 通道是空的就是阻塞, 对发送方不存在这个问题
channel 的源码里面有两个 Goroutine 队列,一个是 接收队列,一个 发送队列。 对于无缓冲的 channel ,一个写操作过来了,会看先看看接收队列有没有 goroutine,如果有的话那么久直接写。如果接收队列没有 goroutine,那么把这个写操作的 goroutine 放到 发送队列。