用两个 goroutine 交替打印 1~100(5种写法)
发布时间 - 2026-01-27 00:00:00 点击率:次用channel实现goroutine交替打印最常用,核心是两个chan struct{}控制执行权:A打印后发信号给chB唤醒B,B打印完再发信号给chA;需初始化chA为带缓冲chan以避免死锁。
用 channel 实现 goroutine 交替打印(最常用)
这是最直观、符合 Go 并发模型的设计。核心是用两个 chan struct{} 控制执行权流转:goroutine A 打印后往 chB 发信号,唤醒 B;B 打印完再发信号给 chA。注意初始必须有一个 goroutine 先拿到“令牌”,否则会死锁。
chA := make(chan struct{}, 1)
chB := make(chan struct{})
chA <- struct{}{} // A 先跑
go func() {
for i := 1; i <= 100; i += 2 {
<-chA
fmt.Println(i)
chB <- struct{}{}
}
}()
go func() {
for i := 2; i <= 100; i += 2 {
<-chB
fmt.Println(i)
chA <- struct{}{}
}
}()
go func() { for i := 1; i <= 100; i += 2 { <-chA fmt.Println(i) chB <- struct{}{} } }() go func() { for i := 2; i <= 100; i += 2 { <-chB fmt.Println(i) chA <- struct{}{} } }()
常见错误:缓冲区设为 0 且没预置信号 → 两个 goroutine 都在等对方,直接 deadlock。别漏掉 chA 这一行。
用 sync.Mutex + sync.Cond 实现交替(适合理解条件等待)sync.Cond 是为“等待某个条件成立”而生的,比纯 sync.Mutex 更精准。这里用一个 sync.Mutex 保护共享变量 turn(标识该谁打印),再用两个 sync.Cond 分别通知 A 和 B。
var mu sync.Mutex
condA := sync.NewCond(&mu)
condB := sync.NewCond(&mu)
turn := 0 // 0: A, 1: B
go func() {
for i := 1; i <= 100; i += 2 {
mu.Lock()
for turn != 0 {
condA.Wait()
}
fmt.Println(i)
turn = 1
condB.Signal()
mu.Unlock()
}
}()
go func() { for i := 1; i <= 100; i += 2 { mu.Lock() for turn != 0 { condA.Wait() } fmt.Println(i) turn = 1 condB.Signal() mu.Unlock() } }()
注意点:必须在 Lock() 后用 for 循环检查条件(不是 if),防止虚假唤醒;Signal() 不保证立刻唤醒,但这里只有两个 goroutine,够用。
用 atomic.Int32 做无锁轮转(轻量但需小心边界)
如果只是交替,不涉及复杂状态,atomic.Int32 足够。定义一个原子计数器,值为 0 表示轮到 A,1 表示轮到 B。每个 goroutine 自旋检查,匹配则打印并切换。
var turn atomic.Int32
turn.Store(0)
go func() {
for i := 1; i <= 100; i += 2 {
for turn.Load() != 0 {
runtime.Gosched() // 主动让出,避免忙等耗 CPU
}
fmt.Println(i)
turn.Store(1)
}
}()
go func() { for i := 1; i <= 100; i += 2 { for turn.Load() != 0 { runtime.Gosched() // 主动让出,避免忙等耗 CPU } fmt.Println(i) turn.Store(1) } }()
性能上比 channel 或 mutex 略高,但要注意:
- 忙等(busy-wait)不加
runtime.Gosched()会吃满一个 CPU 核; -
atomic不能替代锁来保护多字段状态,这里只管单个整数,没问题。
用 select + time.After 实现带超时的交替(防卡死)
纯 channel 方案一旦某 goroutine panic 或提前退出,另一个会永久阻塞在 `select 和 time.After 可做兜底,适用于对稳定性要求高的场景。
chA := make(chan struct{}, 1)
chB := make(chan struct{})
chA <- struct{}{}
go func() {
for i := 1; i <= 100; i += 2 {
select {
case <-chA:
fmt.Println(i)
chB <- struct{}{}
case <-time.After(3 * time.Second):
fmt.Println("A timeout, exiting")
return
}
}
}()
go func() { for i := 1; i <= 100; i += 2 { select { case <-chA: fmt.Println(i) chB <- struct{}{} case <-time.After(3 * time.Second): fmt.Println("A timeout, exiting") return } } }()
适用场景有限——正常逻辑不该超时,但它能帮你快速发现哪个 goroutine 挂了。别滥用,会掩盖真实问题。
用 context.WithCancel 实现可中断的交替(适合长任务)
当打印过程可能被外部取消(比如用户按 Ctrl+C),用 context.Context 最自然。每个 goroutine 监听 ctx.Done(),一收到就退出。
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
go func() {
defer cancel() // A 出错或完成,主动 cancel 整体
for i := 1; i <= 100; i += 2 {
select {
case <-chA:
fmt.Println(i)
chB <- struct{}{}
case <-ctx.Done():
return
}
}
}()
go func() { defer cancel() // A 出错或完成,主动 cancel 整体 for i := 1; i <= 100; i += 2 { select { case <-chA: fmt.Println(i) chB <- struct{}{} case <-ctx.Done(): return } } }()
关键点:cancel 调用要放在 defer 或明确位置,否则可能泄漏;两个 goroutine 都监听同一个 ctx,确保同步退出。
交替逻辑本身简单,但每

# go
# golang
# if
# for
# select
# 循环
# signal
# Struct
# 并发
# channel
# 死锁
# 多字
# 轮到
# 最常用
# 再发
# 这是
# 放在
# 都在
# 发信号
# 令牌
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
北京网站制作公司哪家好一点,北京租房网站有哪些?
大同网页,大同瑞慈医院官网?
Bootstrap整体框架之CSS12栅格系统
javascript中的try catch异常捕获机制用法分析
创业网站制作流程,创业网站可靠吗?
Laravel如何记录自定义日志?(Log频道配置)
Laravel怎么设置路由分组Prefix_Laravel多级路由嵌套与命名空间隔离【步骤】
图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?
北京网站制作的公司有哪些,北京白云观官方网站?
Laravel Fortify是什么,和Jetstream有什么关系
HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】
香港服务器如何优化才能显著提升网站加载速度?
微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】
jQuery validate插件功能与用法详解
弹幕视频网站制作教程下载,弹幕视频网站是什么意思?
Win11关机界面怎么改_Win11自定义关机画面设置【工具】
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
Laravel如何为API编写文档_Laravel API文档生成与维护方法
Laravel怎么实现微信登录_Laravel Socialite第三方登录集成
夸克浏览器网页跳转延迟怎么办 夸克浏览器跳转优化
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
作用域操作符会触发自动加载吗_php类自动加载机制与::调用【教程】
logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?
Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】
Windows家庭版如何开启组策略(gpedit.msc)?(安装方法)
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
网站制作大概要多少钱一个,做一个平台网站大概多少钱?
Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层
如何用美橙互联一键搭建多站合一网站?
Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】
Laravel如何发送邮件和通知_Laravel邮件与通知系统发送步骤
Laravel Sail是什么_基于Docker的Laravel本地开发环境Sail入门
如何在景安云服务器上绑定域名并配置虚拟主机?
,怎么在广州志愿者网站注册?
Python高阶函数应用_函数作为参数说明【指导】
Laravel如何实现全文搜索功能?(Scout和Algolia示例)
七夕网站制作视频,七夕大促活动怎么报名?
Laravel如何使用软删除(Soft Deletes)功能_Eloquent软删除与数据恢复方法
Laravel Seeder填充数据教程_Laravel模型工厂Factory使用
Laravel如何使用Contracts(契约)进行编程_Laravel契约接口与依赖反转
网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?
如何在宝塔面板创建新站点?
Laravel如何清理系统缓存命令_Laravel清除路由配置及视图缓存的方法【总结】
bootstrap日历插件datetimepicker使用方法
Laravel DB事务怎么使用_Laravel数据库事务回滚操作
如何快速搭建高效香港服务器网站?
Laravel如何创建和注册中间件_Laravel中间件编写与应用流程
Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南
Laravel控制器是什么_Laravel MVC架构中Controller的作用与实践
高性能网站服务器配置指南:安全稳定与高效建站核心方案

