美文网首页
go channel

go channel

作者: zcxzcxczcx | 来源:发表于2022-04-15 13:14 被阅读0次

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

相关文章

  • 文章收藏

    channel Go Channel 详解

  • Golang Channel底层实现

    Go Channel 底层实现 目录 channel 是什么 channel 的创建 channel 的发送 ch...

  • 26.Go语言·管道Channel

    main.go model/Channel.go

  • Go语言——channel详解

    Go语言——channel详解 channel和goroutine是go语言最具特色是结构,有必要仔细研究。 源码...

  • channel in Go's runtime

    http://skoo.me/go/2013/09/20/go-runtime-channel/

  • Go channel

    单纯地将函数并发执行是没有意义的,函数与函数之间需要交换数据才能体现并发执行函数的作用。虽然可使用共享内存进行数据...

  • go channel

    [toc] Channel 编译器翻译 关键数据结构 hchan sudog hchan sudog 其中buf ...

  • go channel

    浅析 go channel channel 是 goroutine 之间通信的一种方式,可以类比成 Unix 中的...

  • go channel

    channel channel一般用于goroutine之间的通信;初始化chan :=make(chan int...

  • Go - Channel

    设计理念 执行业务处理的 goroutine 不要通过共享内存的方式通信,而是要通过 Channel 通信的方式分...

网友评论

      本文标题:go channel

      本文链接:https://www.haomeiwen.com/subject/mcatertx.html