【Go程序实现】go如何用两个协程交替打印出123456

在 Go 中,使用两个协程(goroutine)交替打印 123456 的核心在于同步机制流程控制。我们可以借助 channel 来实现两个 goroutine 之间的协作,确保它们按顺序交替打印。


✅ 实现目标

  • 使用两个 goroutine 交替打印数字 1 到 6;
  • 输出顺序为:1 → 2 → 3 → 4 → 5 → 6;
  • 每个 goroutine 负责打印一半的数字(比如,一个打印奇数,一个打印偶数);
  • 打印结束后程序应正确退出。

🛠 实现思路

  1. 共享计数器:使用 atomic 包管理一个共享的计数器变量,确保并发读写安全。
  2. 通道通信:使用两个无缓冲的 channel(aChanbChan)作为“令牌”,控制 goroutine 的执行顺序。
  3. 流程控制
    • 每个 goroutine 在接收到 channel 信号后打印当前数字;
    • 打印完成后,将“令牌”传递给另一个 goroutine;
    • 当数字达到 6 时,两个 channel 被关闭,goroutine 退出循环;
    • 主函数通过 WaitGroup 等待两个 goroutine 完成。

🧩 示例代码

package main

import (
	"fmt"
	"sync/atomic"
)

func main() {
	var counter int32 = 1
	aChan := make(chan bool)
	bChan := make(chan bool)

	var wg sync.WaitGroup
	wg.Add(2)

	// Goroutine A 打印奇数
	go func() {
		defer wg.Done()
		for {
			_, ok := <-aChan
			if !ok {
				return
			}
			current := atomic.LoadInt32(&counter)
			fmt.Print(current, " ")
			newCounter := current + 1
			atomic.StoreInt32(&counter, newCounter)

			if newCounter <= 6 {
				bChan <- true
			} else {
				// 关闭两个 channel,通知所有 goroutine 退出
				close(aChan)
				close(bChan)
			}
		}
	}()

	// Goroutine B 打印偶数
	go func() {
		defer wg.Done()
		for {
			_, ok := <-bChan
			if !ok {
				return
			}
			current := atomic.LoadInt32(&counter)
			fmt.Print(current, " ")
			newCounter := current + 1
			atomic.StoreInt32(&counter, newCounter)

			if newCounter <= 6 {
				aChan <- true
			} else {
				// 关闭两个 channel,通知所有 goroutine 退出
				close(aChan)
				close(bChan)
			}
		}
	}()

	// 启动流程,由 Goroutine A 开始打印 1
	aChan <- true

	// 等待两个 goroutine 完成
	wg.Wait()
}

📌 输出结果

运行程序后,输出如下(顺序严格交替):

1 2 3 4 5 6

🔍 说明

  • atomic.LoadInt32 / atomic.StoreInt32:用于确保共享变量的并发访问安全。
  • aChan / bChan:作为“令牌”控制 goroutine 的执行顺序。每个 goroutine 打印完后将“令牌”传递给对方。
  • close(aChan) / close(bChan):在最后一个数字打印完后关闭 channel,使 goroutine 退出循环,避免阻塞和 goroutine 泄漏。
  • sync.WaitGroup:用于主函数等待两个 goroutine 完成,确保程序不会提前退出。

✅ 小结

通过使用 channel 和原子操作,我们可以安全地实现两个 goroutine 的交替打印,并保证顺序和程序的正确退出。该方法结构清晰、同步机制简单有效,适用于类似的并发控制场景。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值