Channel是Go协程间安全通信的核心机制,通过通信共享内存;无缓冲需同步收发,有缓冲可解耦生产消费;应由发送方关闭且仅一次,关闭后可读尽剩余数据再返回零值和false。
Go 语言中,Channel 是协程(goroutine)之间安全通信的核心机制,不是共享内存,而是“通过通信来共享内存”。用好 Channel,关键在理解其阻塞行为、缓冲策略和关闭语义。
无缓冲 Channel(ch := make(chan int))要求发送与接收必须同步——一方发,另一方必须同时收,否则阻塞。适合精确配对的协作场景,比如任务触发与结果返回。
有缓冲 Channel(ch := make(chan int, 5))像带容量的队列,发送不立即阻塞,直到缓冲满;接收也不阻塞,只要队列非空。适合解耦生产与消费节奏,比如日志收集、批量处理。
常见可靠模式包括:发送-接收配对、扇入(fan-in)、扇出(
fan-out) 和 超时控制。
for range ch)select + 多个 case 或辅助 goroutine 转发)select 配合 time.After(),避免永久阻塞done chan struct{} 传递关闭信号,接收方 select 监听 done 通道Channel 应由发送方关闭,且**只关闭一次**。关闭后不能再发送,否则 panic;但可继续接收,直到所有值读完,之后读取返回零值和 false(ok 为 false)。
不要在接收方关闭,也不要用闭包或多个 goroutine 竞争关闭。常见做法是用单独的 goroutine 控制生命周期,或由主逻辑明确判定发送结束时调用 close(ch)。
v, ok := ,ok 为 false 表示已关闭且无数据
for v := range ch 自动在关闭后退出死锁最常见于:goroutine 启动后只发不收、只收不发,且无其他协程配合;或所有 goroutine 都在等某个未关闭/未发送的 channel。
select 时务必加 default 或 timeout,防止无限等待defer close(ch) 或显式清理逻辑go tool trace 查看 goroutine 阻塞点基本上就这些。Channel 不是万能队列,而是 Go 并发哲学的具象——用可控的阻塞换取清晰的数据流与责任边界。