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

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 设计理念 执行业务处理的 goroutine 不要通过共享内存的方式通信,而是要通过 Channel 通信的方式分...
    kyo1992阅读 629评论 0 0
  • 了解过go的都知道,go最为突出的优点就是它天然支持高并发,但是所有高并发情况都面临着一个很明显的问题,就是并发的...
    GGBond_8488阅读 360评论 0 3
  • channel一个类型管道,通过它可以在goroutine之间发送和接收消息。它是Golang在语言层面提供的go...
    蔡欣圻阅读 13,596评论 4 11
  • 单纯地将函数并发执行是没有意义的,函数与函数之间需要交换数据才能体现并发执行函数的作用。虽然可使用共享内存进行数据...
    JunChow520阅读 444评论 0 2
  • 目录 channel背景 channel基本用法 channel应用场景 channel实现原理 channel数...
    迈莫coding阅读 4,371评论 2 1