senq:等待发送的goroutine队列
recvq:等待接收的goroutine队列
buf是指向底层的循环数组,dataqsiz就是这个循环数组的长度,qcount就是当前循环数组中的元素数量,缓冲的channel才有效。elemsize和elemtype就是我们创建channel时设置的容量大小和元素类型。
sendq、recvq是一个双向链表结构,分别表示被阻塞的goroutine链表,这些 goroutine 由于尝试读取 channel 或向 channel 发送数据而被阻塞。
1.发送数据
1.1有recvq
则从recvq获取一个goroutine (fifo原则)将其唤醒,从send goroutine 直接写另入这个获取到的 goroutine 的栈
1.2没有recvq
1.2.1 channel有缓冲区(dataqsiz>0),
1.2.1.1 缓冲区没满,则把数据拷贝到缓冲区
1.2.1.2 缓冲区满了,如果发送goroutine 非阻塞直接返回,如果发送goroutine 阻塞则将其放入待发送的等待队列sendq,最后调用gopark方法挂起当前的goroutine进入wait状态
1.2.2 channel没有缓冲区,同1、2、1、2
2、接收数据
2.1有senq
2.1.1 没有缓冲区,从senq获取一个goroutine (fifo原则)唤醒, 继续发送数据
2.1.2有缓冲区但缓冲区满了,则从缓冲区接收数据,如果有阻塞的goroutine则唤醒发送数据到channel缓冲区
2.2没有senq
2.2.1 无缓存区,接受goroutine非阻塞则直接返回,阻塞则将其放入待接收的等待队列recvq,最后调用gopark方法挂起当前的goroutine进入wait状态
2.2.2 有缓冲区
2.2.2.1 有缓冲区,有数据 直接接收
2.2.2.1 有缓冲区,无数据 同2.2.1
3、chanel关闭
3.1 一个为nil的channel不允许进行关闭
3.2 不可以重复关闭channel
3.3 获取当前正在阻塞的发送或者接收的goroutine,他们都处于挂起状态,然后进行唤醒。这是发送方不允许在向channel发送数据了,但是不影响接收方继续接收元素,如果没有元素,获取到的元素是零值。使用val,ok := <-ch可以判断当前channel是否被关闭