无缓冲 channel 的发送和接收必须在不同 goroutine 中成对发生,否则会阻塞导致死锁;它像无存水空间的水管,一端发送时另一端必须立即接收。
channel 是 Go 协程间通信的唯一推荐方式,不是“可以选”,而是“应该用”——它把“谁在读、谁在写、数据何时就绪”这些并发细节全收进语言原语里,比手动加锁安全得多,也比共享变量清晰得多。
无缓冲 chan 就像一根没有存水空间的水管:一端开闸(发送),另一端必须立刻接水(接收),否则双方都卡住。这不是 bug,是设计。
常见错误现象:
fatal error: all goroutines are asleep - deadlock!典型场景:主 goroutine 向无缓冲
ch 发送数据,但没起接收协程,或接收协程还没启动。
select + default 做非阻塞试探)ch 再 ,这等于自己等自己
go tool trace 看 goroutine 是否卡在 chan send 或 chan recv 上make(chan int, N) 的 N 不是越大越好,它是背压缓冲区,本质是“允许发送者比接收者快多少”。
性能影响:缓冲区过大

N 通常设为预期并发数 × 1.5~2,比如 10 个 worker 就配 15~20done chan struct{}),用无缓冲更轻量,make(chan struct{}) 即可len(ch) 查当前已存元素数,cap(ch) 查最大容量,二者差值就是还能塞几个close(ch) 只表示“不会再有新数据写入”,但已进队列的数据仍可读完。这是关键区别:关通道 ≠ 清空通道。
常见错误:直接 v := 而不检查是否关闭,导致收到零值(如 0、""、nil)误判为有效数据。
v, ok := ,ok == false 表示通道已关闭且无剩余数据
for range ch 自动读完并退出,但前提是发送端必须调用 close(ch),否则会死锁ch 再执行 ch 会 panic,所以只应在发送端明确知道“发完了”时才关
select 不是轮询,而是同步等待任意一个 case 就绪。如果所有 case 都阻塞,且没写 default,就会永远卡住。
典型陷阱:多个 case 中,某个 ch 已关闭但没做 ok 判断,导致该分支持续返回零值,select 认为它“就绪”,反复选中它。
case 接收时务必用 v, ok := 检查状态,关闭后及时把 ch 设为 nil(nil 的 channel 在 select 中永远不就绪)
,别手写计时器
select 外层套 for 却忘了 break,容易漏掉退出条件Go 的 channel 看似简单,但真正难的是理解“阻塞”背后的调度逻辑——它不是线程挂起,而是 goroutine 被从运行队列摘下,等对方就绪再唤醒。这个细节决定了你写的并发程序是健壮还是脆弱。