golang channel

channel是不同协程之间异步通信的数据结构。

基本用法

1 构造

go 复制代码
ch:=make(chan int)//无缓冲
ch:=make(chan int,10)//有缓冲

2 读操作

go 复制代码
val:=<-ch
<-ch
val,ok:=<-ch

3 写

go 复制代码
var data int
ch<-data

4 关闭

go 复制代码
close(ch)

5 多路复用

go 复制代码
select{
case <-parent.Done():
	child.cancel(false,parent.Err())
case <-child.Done():
}

实现对多个channel同时监听

go 复制代码
select{
case <-ch1://读
	// do some logic
case ch3 <-data://写
	// do some logic
default:
	// 放行
}

核心数据结构

三个核心:

1 并发读写安全,需要锁

2 环形缓冲区(数组+头尾指针),好处是复用数组空间,同时保证内存地址连续

3 承载阻塞goroutine的队列

go 复制代码
type hchan struct{ // runtime包下

	qcount unit  // total data in the queue
	dataqsize uint // size of the circular queue
	buf unsafe.Pointer  // points to an array of dataqsiz elements
	elemsize uint 16
	closed uint 32
	elemtype *_type  //element type
	sendx uint  //send index
	recvx uint  //receive index
	sendq waitq  // list of send waiters
	recvq waitq  //list of recv waiters

	lock mutex
}

hchan:channel数据结构

  • qcount:当前channel中存在多少个元素
  • dataqsize:当前channel中能存放的元素容量
  • buf:channel中用于存放元素的环形缓冲区
  • elemsize:channel中存放元素类型的大小
  • closed:标识channel是否关闭
  • elemtype:channel元素类型
  • sendx:发送元素进入环形缓冲区的index
  • recvx:接收元素所处的环形缓冲区的index
  • sendq:因发送而陷入阻塞的协程队列
  • recvq:因接收而陷入阻塞的协程队列
go 复制代码
type waitq struct{ // 阻塞的协程队列
	first *sudog  // 队列头部
	last *sudog  //队列尾部
}
go 复制代码
type sudog struct{ // 阻塞的协程队列
	g *g
	
	prev *sudog  // 队列头部
	next *sudog  //队列尾部
	elem unsafe.Pointer  // data element(may point to stack)

	isSelect bool

	c *hchan
}
  • sudog:用于包装协程的节点
  • g:goroutine,协程
  • prev:队列中的上一个节点
  • next:队列中的下一个节点
  • isSelect:标识当前协程是否处在select多路复用的流程中
  • c:标识与当前sudog交互的chan
相关推荐
AI玫瑰助手13 分钟前
Python函数:默认参数的定义与注意事项
开发语言·python·信息可视化
油炸自行车21 分钟前
Claude Code 错误:API Error: 400 Failed to deserialize the JSON body into the
开发语言·javascript·json·trae·claude code·api error 400
肩上风骋29 分钟前
C++14特性
开发语言·c++·c++14特性
swipe43 分钟前
Neo4j + Graph RAG 医疗知识图谱工程实践:患者教育问答真正需要的是“关系可追溯”
后端·langchain·llm
源码宝2 小时前
MES系统源码:Java8 + SpringBoot2.7 + MySQL8 + Redis,后端源码清爽易扩展
java·后端·源码·springboot·mes系统·源码二开·mes源码
JAVA社区2 小时前
Java高级全套教程(十)—— SpringCloudAlibaba超详细实战详解
java·开发语言·spring cloud·面试·职场和发展
弥树子2 小时前
踩坑记录:服务器内网调用接口,真实请求URL与官方公开URL不一致问题排查
开发语言·php
金銀銅鐵2 小时前
[Java] 如何理解 class 文件中方法的 descriptor?
java·后端
村口张大爷2 小时前
05 — 分层架构与依赖倒置
后端·架构·系统架构
z落落2 小时前
C# ToCharArray + foreach遍历 + String与StringBuilder
开发语言·c#