go channel

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是否被关闭

你可能感兴趣的:(go channel)