0

0

如何使用Golang实现协程间通信_Golang channel与select使用实践

P粉602998670

P粉602998670

发布时间:2026-01-25 18:16:30

|

200人浏览过

|

来源于php中文网

原创

channel是goroutine间通信的唯一推荐方式,无缓冲channel需收发同步,带缓冲channel可暂存数据;向已关闭channel发送会panic,接收则返回零值;select需default或阻塞操作,否则可能永久阻塞。

如何使用golang实现协程间通信_golang channel与select使用实践

channel 是协程通信的唯一安全通道

Go 中没有共享内存式通信,channelgoroutine 之间传递数据的**唯一推荐方式**。直接读写全局变量或结构体字段会导致竞态(data race),即使加锁也违背 Go “通过通信共享内存” 的设计哲学。

使用 channel 时必须注意:它默认是阻塞的,发送和接收会互相等待;容量为 0(即无缓冲)时,收发必须同时就绪才能完成。

  • make(chan int) 创建无缓冲 channel,适合同步信号(如“任务完成通知”)
  • make(chan string, 10) 创建带缓冲 channel,可暂存 10 个值,避免发送方立即阻塞
  • 向已关闭的 channel 发送会 panic;从已关闭的 channel 接收会立即返回零值 + false

select 多路复用必须配 default 或阻塞操作

select 不是轮询,而是 Go 运行时提供的**非阻塞多路等待机制**。如果所有 case 都不可达(例如所有 channel 都空且无 default),select 会永久阻塞——这是常见卡死原因。

典型误用:select 只有 recv 操作但没 default,而 sender 还没启动或延迟发送。

立即学习go语言免费学习笔记(深入)”;

Beautiful.ai
Beautiful.ai

AI在线创建幻灯片

下载
ch := make(chan int)
// ❌ 卡死:ch 为空,又没 default
select {
case x := <-ch:
    fmt.Println(x)
}
// ✅ 加 default 实现非阻塞尝试
select {
case x := <-ch:
    fmt.Println("received:", x)
default:
    fmt.Println("no data yet")
}

超时控制必须用 time.After 或 context.WithTimeout

不能靠循环 + select + time.Sleep 实现超时,这会浪费 goroutine 和时间精度差。Go 标准做法是把 time.After(d) 当作一个只发一次的 channel 来参与 select

  • time.After(3 * time.Second) 返回 ,3 秒后自动发送当前时间
  • 更健壮的做法是用 context.WithTimeout,尤其在涉及子 goroutine 传播取消信号时
  • 注意:time.After 不可重用,每次超时需新建
ch := make(chan string, 1)
go func() {
    time.Sleep(5 * time.Second)
    ch <- "done"
}()
select {
case msg := <-ch:
    fmt.Println(msg)
case <-time.After(2 * time.Second):
    fmt.Println("timeout!")
}

关闭 channel 前确保所有 sender 已退出

close() 只应由 sender 调用,且**只能关闭一次**。过早关闭会导致 receiver 收到零值并误判为有效数据;重复关闭 panic。

常见模式是用 sync.WaitGroup 等待所有 sender 完成后再关闭:

ch := make(chan int, 10)
var wg sync.WaitGroup
wg.Add(2)
go func() {
    defer wg.Done()
    for i := 0; i < 5; i++ {
        ch <- i
    }
}()
go func() {
    defer wg.Done()
    for i := 5; i < 10; i++ {
        ch <- i
    }
}()
go func() {
    wg.Wait()
    close(ch) // 所有 sender 结束后才关
}()
for v := range ch { // range 自动检测 closed
    fmt.Println(v)
}

真正难的是协调多个 sender 的生命周期,尤其是存在错误提前退出、或需要中途取消的情况——这时候得结合 context 和显式状态检查,不能只依赖 close

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

341

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

209

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

393

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

200

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

192

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

315

2025.06.17

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

55

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4.2万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号