不要通过共享内存来通信,而要通过通信来实现内存共享。
通道是可以让一个goroutine发送特定值到另一个goroutine的通信机制
无缓冲通道:
也称同步通道,无缓冲通道上的发送操作将会阻塞,直到另一个goroutine在对应的通道上执行接收操作,此时传送完成,两个goroutine才能继续执行。反之亦然。
缓冲通道:
缓冲队列有一个缓冲队列,队列的最大长度在创建的时候通过make的容量参数来设置。
缓冲通道上的发送操作在队列的尾部插入一个元素,接收操作从队列的头部移除一个元素。
如果通道满了,发送操作的goroutine会阻塞在通道的等待发送队列中,直到另一个goroutine接收数据。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-TUYushF8-1594887623283)(https://note.youdao.com/yws/res/6858/WEBRESOURCE34805fb12ba1de0185f5488773922bac)]
如果通道为空,接收操作的goroutine会阻塞在通道的等待接收队列中,直到另一个goroutine发送数据。
构造语句,make(chan int)会被golang编译器编译成runtime.makechan函数
函数原型:
func makechan(t *chantype, size int ) *hchan
其中,t *chantype
即构造channel时传入的元素类型。size int
即用户指定的channel缓冲区大小,不指定则为0。该函数的返回值是*hchan
。hchan则是channel在golang中的内部实现
type hchan struct {
qcount uint // buffer中已放入的元素个数
dataqsiz uint // 用户构造channel时指定的buf大小
buf unsafe.Pointer // buffer
elemsize uint16 // buffer中每个元素的大小
closed uint32 // channel是否关闭,== 0代表未closed
elemtype *_type // channel元素的类型信息
sendx uint // buffer中已发送的索引位置 send index
recvx uint // buffer中已接收的索引位置 receive index
recvq waitq // 队列:等待接收的goroutine list of recv waiters
sendq waitq // 队列:等待发送的goroutine list of send waiters
lock mutex // 锁
}
通过分析hchan的属性,得知 buffer和waitq是两个重要的组件,hchan的所有行为都是围绕这两个组件进行
sudog:对当前运行的goroutine和待发送数据的封装,有一个前驱指针和后驱指针hchan的sendq和recvq是由sudog组成的双向链表。
环形缓冲区好处较多,非常适用于FIFO式的固定长度队列
hchan中有两个与buffer相关的变量:recvx和sendx。
从recvx到sendx之间的元素,表示已经正常放入buffer中的数据。
gopark(): 用于协程的切换
gopark函数做的事情分为两点:
//gopark()
func gopark(unlockf func(*g, unsafe.Pointer) bool, lock unsafe.Pointer, reason waitReason, traceEv byte, traceskip int) {
//如果阻塞原因是 Sleep
if reason != waitReasonSleep {
checkTimeouts()
}
mp := acquirem()
gp := mp.curg
status := readgstatus(gp)
//如果状态不是_Grunning
if status != _Grunning && status != _Gscanrunning {
throw("gopark: bad g status")
}
//记录g休眠的原因和上下文
mp.waitlock = lock
mp.waitunlockf = unlockf
gp.waitreason = reason
mp.waittraceev = traceEv
mp.waittraceskip = traceskip
//释放线程m
releasem(mp)
// mcall()作用:
// 1.切换当前线程的堆栈从 g 的堆栈切换到g0的堆栈
// 2.在g0的堆栈上执行新的函数 fn(g)
// 3.保存当前协程的信息,当后续唤醒当前信息时恢复现场
mcall(park_m)
}
//park_m()
func park_m(gp *g) {
//获取g0
_g_ := getg()
if trace.enabled {
traceGoPark(_g_.m.waittraceev, _g_.m.waittraceskip)
}
//更新gp的状态为_Gwaiting
casgstatus(gp, _Grunning, _Gwaiting)
//移除gp和m的绑定关系
dropg()
//进入等待状态前执行前置函数
if fn := _g_.m.waitunlockf; fn != nil {
ok := fn(gp, _g_.m.waitlock)//执行进入wait前的前置函数
_g_.m.waitunlockf = nil
_g_.m.waitlock = nil
//如果waitunlockf函数执行失败,将gp重新置为_Grunnable状态,恢复
if !ok {
if trace.enabled {
traceGoUnpark(gp, 2)
}
casgstatus(gp, _Gwaiting, _Grunnable)
execute(gp, true) // Schedule it back, never returns.
}
}
//发起一轮新的调度
schedule()
}
goready(): 用于唤醒协程
goready主要做的事情:唤醒某一个goroutine,并将该协程转换为runnable状态,并将其放入到P的local queue(本地队列),等待调度wating
func goready(gp *g, traceskip int) {
//切换到g0的栈
systemstack(func() {
ready(gp, traceskip, true)
})
}
func ready(gp *g, traceskip int, next bool) {
if trace.enabled {
traceGoUnpark(gp, traceskip)
}
//获取goroutine的运行状态
status := readgstatus(gp)
//获取g0
_g_ := getg()
//获取m
mp := acquirem() // disable preemption because it can be holding p in a local var
if status&^_Gscan != _Gwaiting {
dumpgstatus(gp)
throw("bad g->status in ready")
}
casgstatus(gp, _Gwaiting, _Grunnable)
// 将g放入P的可运行队列中
runqput(_g_.m.p.ptr(), gp, next)
// 如果有空闲的p 并且没有正在spinning状态的m 则唤醒一个p
if atomic.Load(&sched.npidle) != 0 && atomic.Load(&sched.nmspinning) == 0 {
wakep()
}
releasem(mp)
}
c<-1 对应于runtime中的runtime.chansend函数
memmove() 进行数据的转移,本质上就是一个内存拷贝。
发送数据到通道
/*
c: 通道指针
ep: 指向要发送数据的首地址
block:代表写入操作是否阻塞
*/
func chansend(c *hchan, ep unsafe.Pointer, block bool, callerpc uintptr) bool {
//如果channel为空或者未初始化
if c == nil {
//如果block表示非阻塞,直接return
if !block {
return false
}
//如果block为阻塞,永久阻塞
gopark(nil, nil, waitReasonChanSendNilChan, traceEvGoStop, 2)
throw("unreachable")
}
if debugChan {
print("chansend: chan=", c, "\n")
}
if raceenabled {
racereadpc(c.raceaddr(), callerpc, funcPC(chansend))
}
//如果block为非阻塞
//且channel未关闭
//且(channel非缓冲队列且接收队列receiver为空)或者(channel为有缓冲队列且buf已满 )
//直接return false
if !block && c.closed == 0 && ((c.dataqsiz == 0 && c.recvq.first == nil) ||
(c.dataqsiz > 0 && c.qcount == c.dataqsiz)) {
return false
}
var t0 int64
if blockprofilerate > 0 {
t0 = cputicks()
}
//获取同步锁,保证线程安全
lock(&c.lock)
//如果通道已经关闭,写入数据产生panic
if c.closed != 0 {
unlock(&c.lock)
panic(plainError("send on closed channel"))
}
//*******主要部分*******//
//如果接收队列recv不为空,即有goroutine在接收队列中等待时
//这里不用区分有缓冲和无缓冲channel
//跳过缓冲区,直接将数据发送给等待的接收者goroutine
if sg := c.recvq.dequeue(); sg != nil {
//send函数
send(c, sg, ep, func() { unlock(&c.lock) }, 3)
return true
}
//如果接收队列recv为空
//且缓冲区数据大小 < 通道的大小 (说明此时发送队列sendq一定为空)
if c.qcount < c.dataqsiz {
//直接将数据放入到缓冲区
qp := chanbuf(c, c.sendx)
if raceenabled {
raceacquire(qp)
racerelease(qp)
}
//数据转移:本质上是内存拷贝,ep处拷贝到qp处
typedmemmove(c.elemtype, qp, ep)
c.sendx++
if c.sendx == c.dataqsiz {
c.sendx = 0
}
c.qcount++
unlock(&c.lock)
return true
}
//如果以上条件不满足
//即接收队列为空,且缓冲区buf数据量大小 == 通道大小
//如果block为非阻塞,解锁并返回 false
if !block {
unlock(&c.lock)
return false
}
//如果接收队列recv不为空
//且缓冲队列已满
//则将当前的goroutine加入到sendq队列
gp := getg()
mysg := acquireSudog()
mysg.releasetime = 0
if t0 != 0 {
mysg.releasetime = -1
}
mysg.elem = ep
mysg.waitlink = nil
mysg.g = gp
mysg.isSelect = false
mysg.c = c
gp.waiting = mysg
gp.param = nil
//将当前goroutine的sudog加入到sendq
c.sendq.enqueue(mysg)
//将当前goroutine休眠
gopark(chanparkcommit, unsafe.Pointer(&c.lock), waitReasonChanSend, traceEvGoBlockSend, 2)
KeepAlive(ep)
if mysg != gp.waiting {
throw("G waiting list is corrupted")
}
gp.waiting = nil
gp.activeStackChans = false
if gp.param == nil {
if c.closed == 0 {
throw("chansend: spurious wakeup")
}
panic(plainError("send on closed channel"))
}
gp.param = nil
if mysg.releasetime > 0 {
blockevent(mysg.releasetime-t0, 2)
}
mysg.c = nil
releaseSudog(mysg)
return true
}
send函数
// send 函数处理向一个空的 channel 发送操作
//直接拷贝的作用:绕开了缓冲区,减少一次加锁操作,提高性能
// ep 指向被发送的元素,会被直接拷贝到接收的 goroutine
// 之后,接收的 goroutine 会被唤醒
// c 必须是空的(因为等待队列里有 goroutine,肯定是空的)
// ep 必须是非空,并且它指向堆或调用者的栈
func send(c *hchan, sg *sudog, ep unsafe.Pointer, unlockf func(), skip int) {
// 省略一些用不到的
// ……
// sg.elem 指向接收到的值存放的位置,如 val <- ch,指的就是 &val
if sg.elem != nil {
// 直接拷贝内存(从发送者到接收者)
sendDirect(c.elemtype, sg, ep)
sg.elem = nil
}
// sudog 上绑定的 goroutine
gp := sg.g
// 解锁
unlockf()
gp.param = unsafe.Pointer(sg)
if sg.releasetime != 0 {
sg.releasetime = cputicks()
}
// 唤醒接收的 goroutine. skip 和打印栈相关,暂时不理会
goready(gp, skip+1)
}
简单来说,向通道中发送数据的整个流程如下:
1.检查recvq是否为空。如果不为空,则从recvq头部取一个goroutine,将数据发送过去,并唤醒对应的goroutine即可。
2.如果recv为空,且缓冲取未满,则将数据放入channel buffer缓冲区中。
3.如果buffer已满,则将要发送的数据和当前goroutine打包成sudog对象放入到sendq中,并将goroutine置为wating状态
注意: channel的整个发送过程和接收过程都使用了runtime.mutex进行加锁。runtime.mutex是runtime相关源码中常用的一个轻量级锁。
/*
block:表示当channel无法返回数据时是否阻塞等待 比如: 当block为false并且channel中没有数据时,直接返回
*/
func chanrecv(c *hchan, ep unsafe.Pointer, block bool) (selected, received bool) {
//*******前置场景*******//
if debugChan {
print("chanrecv: chan=", c, "\n")
}
//如果通道为空或未初始化
if c == nil {
//如果block为非阻塞
//直接返回
if !block {
return
}
//如果block为阻塞,调用gopark()阻塞当前goroutine
gopark(nil, nil, waitReasonChanReceiveNilChan, traceEvGoStop, 2)
throw("unreachable")
}
//如果通道不为空或者已经初始化
//如果block为非阻塞接收
//且((通道为无缓冲通道且发送对列为空) 或者 (通道为有缓冲通道且缓冲区元素为0且通道未关闭))
//直接return false false
if !block && (c.dataqsiz == 0 && c.sendq.first == nil ||
c.dataqsiz > 0 && atomic.Loaduint(&c.qcount) == 0) &&
atomic.Load(&c.closed) == 0 {
return
}
var t0 int64
if blockprofilerate > 0 {
t0 = cputicks()
}
//获取全局锁
lock(&c.lock)
//*******主要部分*******//
//如果通道已经关闭
//且缓冲区中无元素
//直接返回true和false(非正常返回)
if c.closed != 0 && c.qcount == 0 {
if raceenabled {
raceacquire(c.raceaddr())
}
unlock(&c.lock)
if ep != nil {
//返回空值
//typedmemclr根据类型清理相应的地址的内存
typedmemclr(c.elemtype, ep)
}
return true, false
}
//如果等待发送对列sendq不为空 (注意:此时的缓冲区一定是满的)
//有可能是
//1.非缓冲型channel 则直接进行内存拷贝
//2.缓冲型channel,但buf满了, 则接收buf头部的元素,并将发送队列头goroutine的元素放到循环数组的尾部
//返回true,true
if sg := c.sendq.dequeue(); sg != nil {
recv(c, sg, ep, func() { unlock(&c.lock) }, 3)
return true, true
}
//如果等待发送队列sendq为空
//且通道非空 (注意: 一定是缓冲型channel)
//则从通道中获取数据
if c.qcount > 0 {
//从循环数组里找到要接收的元素
qp := chanbuf(c, c.recvx)
if raceenabled {
raceacquire(qp)
racerelease(qp)
}
//ep不为nil,说明未忽略要接收的值,即val<-ch,非<-ch
if ep != nil {
//内存拷贝
typedmemmove(c.elemtype, ep, qp)
}
//清理循环数组recvx处的值
typedmemclr(c.elemtype, qp)
c.recvx++
if c.recvx == c.dataqsiz {
c.recvx = 0
}
c.qcount--
unlock(&c.lock)
return true, true
}
//如果等待发送队列sendq为空
//且通道中元素个数为0,通道为空
//且block为非阻塞
//则直接返回false,false
if !block {
unlock(&c.lock)
return false, false
}
//如果以上情况均不满足,即
//等待发送队列sendq为空
//且通道为空
//且block为阻塞
//则将该goroutine阻塞
gp := getg()
mysg := acquireSudog()
mysg.releasetime = 0
if t0 != 0 {
mysg.releasetime = -1
}
//保存待接收变量的地址ep
mysg.elem = ep
mysg.waitlink = nil
gp.waiting = mysg
mysg.g = gp
mysg.isSelect = false
mysg.c = c
gp.param = nil
//goroutine加入等待接收队列
c.recvq.enqueue(mysg)
gopark(chanparkcommit, unsafe.Pointer(&c.lock), waitReasonChanRece sg.elem = nil
ive, traceEvGoBlockRecv, 2)
// 如果被唤醒了,继续执行扫尾工作
if mysg != gp.waiting {
throw("G waiting list is corrupted")
}
gp.waiting = nil
gp.activeStackChans = false
if mysg.releasetime > 0 {
blockevent(mysg.releasetime-t0, 2)
}
closed := gp.param == nil
gp.param = nil
mysg.c = nil
releaseSudog(mysg)
return true, !closed
}
typedmemmove(c.elemtype, ep, qp)
表示buffer中的当前可读元素拷贝到接收变量的地址处。typedmemmove(c.elemtype, qp, sg.elem)
表示将sendq中goroutine等待发送的数据拷贝到buffer中。因为此后进行了recv++
, 因此相当于把sendq中的数据放到了队尾。func recv(c *hchan, sg *sudog, ep unsafe.Pointer, unlockf func(), skip int) {
// 如果是非缓冲型的 channel
if c.dataqsiz == 0 {
if raceenabled {
racesync(c, sg)
}
// 未忽略接收的数据
if ep != nil {
// 直接拷贝数据,从 sender goroutine -> receiver goroutine
recvDirect(c.elemtype, sg, ep)
}
} else {
// 缓冲型的 channel,但 buf 已满。
// 将循环数组 buf 队首的元素拷贝到接收数据的地址
// 将发送者的数据入队。实际上这时 recvx 和 sendx 值相等
// 找到接收游标
qp := chanbuf(c, c.recvx)
// 将接收游标处的数据拷贝给接收者
if ep != nil {
typedmemmove(c.elemtype, ep, qp)
}
// 将发送者数据拷贝到 buf
typedmemmove(c.elemtype, qp, sg.elem)
// 更新游标值
c.recvx++
if c.recvx == c.dataqsiz {
c.recvx = 0
}
c.sendx = c.recvx
}
sg.elem = nil
gp := sg.g
// 解锁
unlockf()
gp.param = unsafe.Pointer(sg)
if sg.releasetime != 0 {
sg.releasetime = cputicks()
}
// 唤醒发送的 goroutine。需要等到调度器的光临
goready(gp, skip+1)
}
简单来说,向通道中发送数据的整个流程如下:
1.检查sendq是否为空。如果不为空,则从sendq头部取一个goroutine,将缓冲区中队首的元素拷贝给接收变量,同时将sendq中的元素拷贝到队尾。
2.如果sendq为空,且缓冲取不为空,则直接从缓冲取队首获取元素。
3.如果缓冲区为空,则将当前goroutine打包成sudog对象放入到recvq中,并将goroutine置为wating状态
简单来说,这里channel将buffer中队首的数据拷贝给了对应的接收变量,同时将sendq中的元素拷贝到了队尾,这样才可以做到数据的FIFO(先入先出)。
接下来可能有点绕,c.sendx = c.recvx
, 这句话实际的作用相当于c.sendx = (c.sendx+1) % c.dataqsiz
,因为此时buffer依然是满的,所以sendx == recvx
是成立的。