Go语言中channel是协程通信与同步的核心抽象通道,用于任务分发、结果收集和并发控制;典型用法包括带缓冲channel分发任务、for range收集结果并配合WaitGroup确保安全关闭,以及用容量受限channel实现限流。
Go 语言中,channel 是协程(goroutine)间通信与同步的核心机制,不是用来共享内存的“管道”,而是承载数据传递、控制流协调和状态同步的抽象通道。用好 channel,关键不在于“传什么”,而在于“何时传、谁等谁、传完怎么继续”。
典型场景:启动多个 goroutine 并行处理一批任务,主协程等待全部完成并汇总结果。
建议使用带缓冲的 channel 分发任务(避免发送阻塞),再用无缓冲或带缓冲 channel 收集结果,并配合 sync.WaitGroup 或 for range + 关闭 channel 来判断完成:
jobs := make(chan int, 10) 和 results := make(chan int, 10)
for range results 自动等待所有结果送达避免瞬间启动成百上千 goroutine 压垮系统,可用 channel 作“信号量”来限制活跃协程数。
做法很简单:创建一个容量为 N 的 channel,比如 sem := make(chan struct{}, 3)。每个 goroutine 启动前先向 sem 发送一个空结构体(获取许可),处理完再接收一次(释放许可):
sem —— 若已满则阻塞,自然实现排队 —— 归还许可,让下一个协程进入
CSP 思想从 channel 接收数据时,若发送方永不发送,接收方会一直挂起。用 select 配合 time.After 可设置超时,让协作更健壮。
select { case res :=
default 分支实现非阻塞尝试(类似“if channel not empty then read”)关闭 channel 表示“不会再有新数据发送”,仅应由发送方关闭,且只能关一次。接收方通过 v, ok := 中的 ok 判断是否已关闭。
ok == false,说明 channel 已关且缓冲区清空,可安全退出循环