用两个 goroutine 交替打印 1~100(5种写法)
技术百科
冷炫風刃
发布时间:2026-01-27
浏览: 次 用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 最自然。每个 goroutin

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 } } }()
关键点:cancel 调用要放在 defer 或明确位置,否则可能泄漏;两个 goroutine 都监听同一个 ctx,确保同步退出。
交替逻辑本身简单,但每种写法背后对应不同工程诉求:channel 通用,Cond 适合教学,atomic 要防忙等,select+timeout 是防御性编程,context 是面向生命周期管理。选哪种,取决于你真正想解决的问题,而不是“看起来酷不酷”。
# 放在
# 这是
# 都在
# 令牌
# 多字
# 最常用
# go
# golang
# 循环
# 并发
# if
# signal
# 死锁
# Struct
# channel
# select
# for
# 发信号
# 再发
# 轮到
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- 如何高效删除 NumPy 二维数组中所有元素相同的
- c++ std::atomic如何保证原子性 c+
- Win11如何添加/删除输入法 Win11切换中英
- Golang如何实现基本的用户注册_Golang用
- PHP主流架构怎么集成Redis缓存_配置步骤【方
- mac怎么退出id_MAC退出iCloud账号与A
- php8.4匿名类怎么用_php8.4匿名类创建与
- Win11开始菜单打不开_修复Windows 11
- php8.4新语法match怎么用_php8.4m
- 如何在Golang中修改数组元素_通过指针实现原地
- 如何用正则与预处理结合精准拦截拼接式垃圾域名
- Win11 explorer.exe频繁崩溃_修复
- c++ nullptr与NULL区别_c++11空
- C++如何获取CPU核心数?(std::threa
- Python技术债务管理_长期维护解析【教程】
- php订单日志怎么记录物流_php记录订单物流变更
- 如何高效获取循环末次生成的 NumPy 数组最后一
- Python 模块的 __name__ 属性如何由
- Mac如何开启夜览模式_Mac护眼模式设置与定时
- PHP的FastAdmin架构适合二次开发吗_特点
- 如何理解Go指针和内存分配关系_Go Pointe
- mac怎么分屏_MAC双屏显示与分屏操作技巧【指南
- Windows10系统怎么查看硬盘健康_Win10
- MAC如何修改默认应用程序_MAC文件后缀关联设置
- PHP的Workerman对架构扩展有啥帮助_应用
- Python项目回滚策略_发布安全说明【指导】
- MAC如何启用访达侧边栏显示_MAC Finder
- Win11如何设置省电模式 Win11开启电池节电
- Python解释执行模型_字节码流程说明【指导】
- php8.4如何配置ssl证书_php8.4htt
- 手机php怎么转mp4_手机端php文件转mp4a
- 如何在Golang中实现微服务服务拆分_Golan
- TestNG的testng.xml配置文件怎么写
- 如何用::实现单例模式_php静态方法与作用域操作
- Python与OpenAI接口集成实战_生成式AI
- 如何使用Golang处理网络超时错误_Golang
- Win10如何卸载Skype_Win10卸载Sky
- 如何使用Golang sort排序切片_Golan
- Win11输入法选字框不见了怎么办_Win11输入
- php中常量能用::访问吗_类常量与作用域操作符使
- 如何在 Django 中安全修改用户密码而不使会话
- Golang如何遍历目录文件_Golang fil
- SAX解析器是什么,它与DOM在处理大型XML文件
- Win10路由器怎么隐藏ssid Win10隐藏w
- Windows11如何设置专注助手_Windows
- Linux如何使用grep搜索文件内容_Linux
- Win11如何设置开机自动联网 Win11宽带连接
- 企业SEO优化选择网站建设模板的技巧
- Win11截图快捷键是什么_Win11自带截图工具
- php订单日志怎么按金额排序_php按订单金额排序

QQ客服