很多人写 Go 后端时都会用 channel。
任务分发要用它,worker pool 要用它,超时控制要配合 select,优雅退出常常是 done chan struct{},限流时又会拿 buffered channel 当信号量。
但真的遇到的时候,很多人一碰到下面这些问题就开始发虚:
nil channel 为什么会永远阻塞?close 之后到底还能不能继续读?v, ok := <-ch 里的 ok=false 到底什么时候出现?select 为什么看起来简单,runtime 实现却明显更复杂?如果面对这些问题时并不是胸有成竹,说明你对 channel 的理解,大概率还停留在“会用语法”这一步。
这篇文章我不打算只讲语法糖,而是顺着一条更实用的线讲清楚:
hchan、等待队列、唤醒逻辑到底怎么配合。只会写下面这种代码,其实不算真正理解 channel:
ch := make(chan int, 10) ch <- 1 v := <-ch _ = v
真正的难点从来不是“怎么写”,而是“它在什么状态下会阻塞、什么时候会 panic、为什么 close 可以做广播、为什么有些 goroutine 会莫名其妙泄漏”。
Go 后端里,channel 一般出现在这几类地方:
这些场景背后,其实都不是“单纯传个值”那么简单,而是在依赖 channel 的同步语义和调度行为。
所以如果你只记住“channel 是管道”,其实是远远不够的。
你还得知道它什么时候像队列,什么时候像同步握手,什么时候像广播器,什么时候又会把 goroutine 卡死在原地…
如果只用一句话概括 channel,我会这么讲:
对外,channel 是带类型的通信管道;对内,它是锁 + 环形缓冲区 + 等待队列 + 唤醒逻辑。
这句话非常重要,因为它同时解释了两层东西。
第一层是语言语义:
你可以发送、接收、关闭、range、select,这些都是 Go 语言承诺给你的可用行为。
第二层是底层实现:
runtime 为了把这些语义落地,需要去维护:
sendq。recvq。这也是为什么你表面上看到的是 ch <- x 和 <-ch,但实际发生的是一整套状态判断和调度行为。
较真的家伙,可以具体了解一下:后续还会在细讲,这张图可以先略微看下

理解 channel,最先要记住的不是源码,而是状态。
我建议可以先把这 4 种状态背下来:
| 状态 | 发送 | 接收 | close |
|---|---|---|---|
nil channel | 永远阻塞 | 永远阻塞 | panic |
| 无缓冲 channel | 必须等接收方 ready | 必须等发送方 ready | 可以关闭 |
| 有缓冲 channel | buffer 未满可直接发送 | buffer 非空可直接接收 | 可以关闭,剩余数据仍可读 |
| 已关闭且已空 | panic | 立刻返回零值,ok=false | 重复 close panic |
这张表之所以重要,平时我们项目遇到的,90%都源于此。
永远阻塞,却在 select 里很好用
未初始化的 channel 零值就是 nil。
这种行为非常的 “绝”:
close(nil) 会直接 panic。var ch chan int // ch <- 1 // 永久阻塞 // <-ch // 永久阻塞 // close(ch) // panic
第一次了解到的时候是非常疑惑的,
因为这种特性挺直觉的,因为只是一个没初始化的值!为啥会好用呢?
其实,是因为可以通过select将其玩通。
因为把某个 case 对应的 channel 变量设成 nil,就等于临时禁用这个分支。
var in <-chan int
for {
select {
case v := <-in:
_ = v
default:
return
}
}
如果运行过程中把 in = nil,那么这个 case v := <-in 就永远不会被选中。这个技巧在状态切换、阶段性关闭某条分支时非常顺手。
它通常被用作同步状态,撇开了对队列的幻想
很多人对无缓冲 channel 的第一理解是“没有 buffer”。
这没错,但不够准。
更准确的说法是:无缓冲 channel 的核心是发送和接收必须配对完成。
ch := make(chan int)
go func() {
ch <- 42
}()
v := <-ch
fmt.Println(v)
这段代码里,发送方执行 ch <- 42 后,如果接收方还没 ready,就会阻塞;接收方执行 <-ch 后,如果发送方还没 ready,也会阻塞。
所以无缓冲 channel 本质上是一种同步握手。
它特别适合表达“我不是想排队,我就是要等对方真的接住”。
它是固定容量的环形队列
有缓冲 channel 会在 runtime 里维护一个固定容量的环形缓冲区。
ch := make(chan int, 8)
它的行为可以简单记成 4 句话:
sendq。recvq。从值的角度看,它可以理解为 FIFO 队列。

但这里有个很容易被忽略的点:FIFO 不等于 goroutine 调度绝对公平。
也就是说,channel 内部的元素顺序可以按先入先出来理解,但多个 goroutine 谁先被调度到、谁先抢到执行机会,不是你靠 channel 就能绝对保证的。
不是销毁,而是告诉接收方“不会再有新值了”
close(ch) 最容易被误解成“把 channel 删除了”。
其实不是。
关闭的语义更像是一句广播声明:
发送方结束了,后面不会再有新值。
关闭之后要分两种情况看:
ok=false。ch := make(chan int, 2) ch <- 1 ch <- 2 close(ch) fmt.Println(<-ch) // 1 fmt.Println(<-ch) // 2 v, ok := <-ch fmt.Println(v, ok) // 0 false
而另外两件事一定会 panic:
go官方推荐的优雅做法是,发送方主动close,而非接受方进行close
channel 不只是传值工具,它还是同步原语
这一部分是很多开发者容易忽略,但却很重要的点
channel 的意义不只是“传个数据过去”,还包括:
建立 happens-before 关系。
也就是说,某次发送或者关闭之前发生的写操作,对后续被唤醒的接收方来说是可见的。
看一个最经典的例子:
var s string
done := make(chan struct{})
go func() {
s = "hello"
close(done)
}()
<-done
fmt.Println(s) // 一定能看到 hello
这里真正关键的不是 done 里有没有值,而是:
s = "hello"。close(done)。<-done 之后继续往下走。这个顺序能成立,不是靠“碰巧”,而是因为 Go 内存模型明确给了你同步保证。
所以很多时候,channel 传递的不是数据,而是“某件事已经发生”的事实。
这也是为什么 done chan struct{} 这种写法在 Go 里这么常见:它利用的本质是同步语义,不是数据语义。
hchan、waitq、sudog
如果继续往底层看,channel 在 runtime 里的核心结构叫 hchan。
虽然没必要记完整源码,但下面这些字段最好了解:
| 字段 | 作用 |
|---|---|
qcount | 当前缓冲区里的元素个数 |
dataqsiz | 缓冲区容量 |
buf | 指向环形缓冲区 |
sendx | 下一次写入的位置 |
recvx | 下一次读取的位置 |
recvq | 等待接收的 goroutine 队列 |
sendq | 等待发送的 goroutine 队列 |
closed | channel 是否已关闭 |
lock | 保护以上状态的互斥锁 |
可以把它理解成一个简化版结构:
type hchan struct {
qcount uint
dataqsiz uint
buf unsafe.Pointer
sendx uint
recvx uint
recvq waitq
sendq waitq
closed uint32
lock mutex
}
这里还有个很关键的角色:sudog。
很多人第一次看到这个名字会觉得奇怪,但它解决的问题其实很现实:
goroutine 和 channel 不是一对一关系。
尤其是 select 里,一个 goroutine 可能同时等多个 channel;反过来,一个 channel 也可能同时被很多 goroutine 等待。
所以 runtime 不能简单在 goroutine 或 channel 上只挂一个指针,而是需要一个“等待记录单元”把两边串起来,这个记录单元就是 sudog。
你可以把它理解成:
某个 goroutine 正在某个 channel 上等待一次发送或接收。
这一段不建议死记 runtime 代码,没啥意义。
所以会通过决策顺序来给大家描绘。
我把发送流程,模拟成下面这棵决策树:
nil:则是:永久阻塞。
则是:panic。
则是:直接把值交给接收者,必要时绕过 buffer。
则是:写入环形缓冲区。
当前 goroutine 封装成 sudog,进入 sendq,阻塞等待。
切记
即使是 buffered channel,只要此时已经有接收者在等,发送也可能直接把值交给接收者,而不是先进 buffer。
所以不要把 channel 想得太机械,好像任何值都必须先排进缓冲区。
接收的逻辑和发送基本对称,但对 closed 的处理更特殊:
nil:则是:永久阻塞。
则是:立刻返回零值,ok=false。
则是:和发送者直接配对,必要时与 buffer 做一次交接。
则是:直接从环形缓冲区读取。
当前 goroutine 进入 recvq,阻塞等待发送者唤醒。
所以 ok=false 的真正含义不是“这次接收失败了”,而是:
channel 已经关闭,并且已经没有剩余数据了。
v, ok := <-ch
if !ok {
// channel 已关闭且读空
}
close 的本质非常适合一句话记忆:
设置关闭标记,然后广播唤醒所有等待者。
它不是销毁 channel,而是做 3 件事:
nil 和重复关闭。closed 标记设为 1。等待中的接收者被唤醒后:
ok=false。等待中的发送者被唤醒后:
这也是为什么向已关闭 channel 发送是非常危险的。
语言层面上,select 规则不复杂:
default,走 default。但 runtime 难点在于:
一个 goroutine 同时等多个 channel,但最终只能有一个 case 赢。
所以实现时通常要做这些事:
sudog 的形式挂到多个 channel 的等待队列。所以 select 语义不难,复杂度主要都在 runtime 的并发协调上。
讲到底层,不是为了背源码,而是为了更好的服务项目。
原因很简单:发送方最清楚“后面还有没有数据”。
如果由接收方随手关闭 channel,很容易导致另一个发送方还没停,结果下一次发送直接 panic。
单发送者场景下,发送方自己关闭,最简单也最安全。
多发送者场景下,最好由协调者统一关闭,比如:
sync.Once。这是一个非常常见的坑。
len(ch) 只是某一瞬间的快照,不是同步保证。
if len(ch) > 0 {
v := <-ch
_ = v
}
这段代码的问题在于:你检查完长度,到真正接收之间,别的 goroutine 完全可能已经把 channel 清空了。
这就是典型的 TOCTOU 问题。
更稳的写法一般是:
select {
case v := <-ch:
_ = v
default:
// 暂时没数据
}
这是线上最容易踩的一个大坑。
如果消费者退出了,但生产者还在不停往 channel 发数据,生产者 goroutine 很可能永久阻塞在发送上。
这些 goroutine 不会自动被 GC 回收,因为它们还“活着”,只是卡住了。
解决思路通常有两种:context 或 done channel。
func producer(ctx context.Context, ch chan<- int) {
for {
select {
case <-ctx.Done():
return
case ch <- getValue():
}
}
}
或者:
done := make(chan struct{})
go func() {
for {
select {
case <-done:
return
case ch <- getValue():
}
}
}()
// 需要停止时
close(done)
这类写法在后端里非常常见,而且很好用。
limit := make(chan struct{}, 3) // 最多 3 个并发
for _, job := range jobs {
job := job
go func() {
limit <- struct{}{} // acquire
defer func() { <-limit }() // release
do(job)
}()
}
本质上,这就是拿 buffered channel 充当一个计数信号量。
这是我最想强调的一句判断:
channel 更擅长表达协作流程和所有权转移,mutex 更擅长保护共享状态。
如果你的问题本质上是:
那 channel 往往很合适。
如果你的问题本质上只是:
map。struct。那 sync.Mutex 往往更直接,成本也更低。
不要为了“Go 很推崇 channel”就到处硬用。
锁 + 环形缓冲区 + sendq/recvq + 唤醒逻辑。nil channel 发送和接收都会永久阻塞,close(nil) 会 panic。close 不是销毁,而是广播“不会再有新值”。v, ok := <-ch 里,ok=false 只会在“channel 已关闭且已空”时出现。happens-before。hchan 的关键字段要知道:qcount、dataqsiz、buf、sendx、recvx、recvq、sendq、closed、lock。sudog 记录的是“某个 goroutine 正在某个 channel 上等待一次发送或接收”。select 的复杂度来自“一个 goroutine 同时等多个 channel,但最终只能有一个 case 赢”。close,不要靠 len(ch) 做同步判断,下游不消费时上游必须可停。最后还是回到最实用的那句话。
真正理解 channel,不是会写 make(chan T),而是你已经知道它什么时候像队列,什么时候像同步点,什么时候像广播器,什么时候会把 goroutine 永远卡住。
这一步迈过去,Go 并发编程才算真正入门。
1、菜鸟教程(channel)
2、go语言中文文档(channel)