0

0

Go语言中并发临界区交替执行的优雅实现:基于双通道模式

碧海醫心

碧海醫心

发布时间:2025-10-29 15:55:27

|

292人浏览过

|

来源于php中文网

原创

Go语言中并发临界区交替执行的优雅实现:基于双通道模式

本文探讨go语言中如何确保两个或多个并发goroutine的临界区代码段严格交替执行。通过引入“双通道”模式,每个goroutine拥有一个接收通道和一个发送通道,形成一个信号传递的闭环,有效控制临界区的执行顺序,实现精确的交替调度,并具备良好的扩展性,是处理此类并发同步问题的简洁高效方案。

并发编程中,我们经常需要协调不同Goroutine的执行顺序,尤其是在涉及共享资源或特定业务逻辑时。有时,我们不仅需要确保临界区互斥执行,更要求它们严格按照特定的顺序交替执行,例如:临界区A执行后必须是临界区B,B执行后又必须是A,如此往复。Go语言提供了强大的并发原语,其中通道(channel)是实现这种精细控制的理想工具

并发临界区交替执行的需求

假设我们有两个Goroutine f1 和 f2,它们各自包含一个临界区代码段(CS1和CS2)。我们的目标是确保这两个临界区始终交替执行:CS1 -> CS2 -> CS1 -> CS2 ...。传统的互斥锁(sync.Mutex)只能保证临界区不会同时被多个Goroutine访问,但无法强制执行顺序。要实现严格的交替执行,我们需要一种机制来让Goroutine在完成自己的临界区后,明确地“通知”下一个Goroutine开始执行其临界区。

双通道机制:原理与设计

解决这种交替执行问题的核心思想是构建一个“令牌传递”系统,我们称之为“双通道模式”。每个参与交替执行的Goroutine都拥有两个通道:

  1. 接收通道 (do channel): 用于接收“执行令牌”,表示轮到该Goroutine执行其临界区了。
  2. 发送通道 (next channel): 用于在完成临界区后,将“执行令牌”传递给下一个Goroutine。

通过这种设计,Goroutine在进入临界区前会尝试从其接收通道获取令牌。如果通道为空,它将阻塞,直到有令牌到来。一旦获取令牌并完成临界区,它会将令牌发送到下一个Goroutine的接收通道,从而激活下一个Goroutine。这就像传递一个“接力棒”,确保每次只有一个Goroutine持有接力棒(即执行令牌),并按照预设的顺序传递。

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

代码实现与解析

下面通过一个具体的Go语言示例来展示如何实现双通道模式。

万兴爱画
万兴爱画

万兴爱画AI绘画生成工具

下载

Goroutine函数设计 (f1, f2)

每个Goroutine函数需要接收两个通道参数:一个用于接收令牌(do),另一个用于发送令牌(next)。

package main

import (
    "fmt"
    "time"
)

// f1 包含临界区1,并在完成后将令牌传递给f2
func f1(do chan bool, next chan bool, id int) {
    for i := 0; i < 3; i++ { // 循环执行几次以观察交替效果
        // ... some code before critical section 1
        fmt.Printf("Goroutine %d: Before CS1\n", id)

        <-do // 等待接收令牌,表示轮到f1执行CS1

        // critical section 1 (CS1)
        fmt.Printf("Goroutine %d: Executing CS1 (Iteration %d)\n", id, i+1)
        time.Sleep(100 * time.Millisecond) // 模拟临界区工作
        // end critical section 1

        next <- true // 将令牌发送给下一个Goroutine (f2)
        fmt.Printf("Goroutine %d: After CS1, passed token\n", id)

        // ... more code after critical section 1
    }
}

// f2 包含临界区2,并在完成后将令牌传递给f1
func f2(do chan bool, next chan bool, id int) {
    for i := 0; i < 3; i++ { // 循环执行几次以观察交替效果
        // ... some code before critical section 2
        fmt.Printf("Goroutine %d: Before CS2\n", id)

        <-do // 等待接收令牌,表示轮到f2执行CS2

        // critical section 2 (CS2)
        fmt.Printf("Goroutine %d: Executing CS2 (Iteration %d)\n", id, i+1)
        time.Sleep(100 * time.Millisecond) // 模拟临界区工作
        // end critical section 2

        next <- true // 将令牌发送给下一个Goroutine (f1)
        fmt.Printf("Goroutine %d: After CS2, passed token\n", id)

        // ... more code after critical section 2
    }
}

主函数调度 (main)

在 main 函数中,我们需要创建两个带缓冲的通道,并初始化第一个Goroutine的接收通道,使其能够率先启动。

func main() {
    // 创建两个带缓冲的通道,缓冲大小为1,确保每次只有一个令牌在流通
    cf1 := make(chan bool, 1) // f1的接收通道,f2的发送通道
    cf2 := make(chan bool, 1) // f2的接收通道,f1的发送通道

    // 初始时,将一个令牌放入cf1,让f1能够首先启动其临界区
    cf1 <- true

    // 启动两个Goroutine
    go f1(cf1, cf2, 1) // f1 接收cf1的令牌,完成后将令牌发送到cf2
    go f2(cf2, cf1, 2) // f2 接收cf2的令牌,完成后将令牌发送到cf1

    // 为了防止main Goroutine过早退出,导致子Goroutine无法完成,
    // 我们需要一个机制来等待。这里使用select{}来阻塞main Goroutine,
    // 实际应用中可能使用sync.WaitGroup或特定的退出信号。
    select {}
}

完整示例代码

package main

import (
    "fmt"
    "time"
)

// f1 包含临界区1,并在完成后将令牌传递给f2
func f1(do chan bool, next chan bool, id int) {
    for i := 0; i < 3; i++ { // 循环执行几次以观察交替效果
        fmt.Printf("Goroutine %d: Waiting for token to execute CS1\n", id)
        <-do // 等待接收令牌,表示轮到f1执行CS1

        // critical section 1 (CS1)
        fmt.Printf("Goroutine %d: Executing CS1 (Iteration %d)\n", id, i+1)
        time.Sleep(100 * time.Millisecond) // 模拟临界区工作
        // end critical section 1

        next <- true // 将令牌发送给下一个Goroutine (f2)
        fmt.Printf("Goroutine %d: Finished CS1, passed token to next\n", id)
    }
}

// f2 包含临界区2,并在完成后将令牌传递给f1
func f2(do chan bool, next chan bool, id int) {
    for i := 0; i < 3; i++ { // 循环执行几次以观察交替效果
        fmt.Printf("Goroutine %d: Waiting for token to execute CS2\n", id)
        <-do // 等待接收令牌,表示轮到f2执行CS2

        // critical section 2 (CS2)
        fmt.Printf("Goroutine %d: Executing CS2 (Iteration %d)\n", id, i+1)
        time.Sleep(100 * time.Millisecond) // 模拟临界区工作
        // end critical section 2

        next <- true // 将令牌发送给下一个Goroutine (f1)
        fmt.Printf("Goroutine %d: Finished CS2, passed token to next\n", id)
    }
}

func main() {
    // 创建两个带缓冲的通道,缓冲大小为1,确保每次只有一个令牌在流通
    cf1 := make(chan bool, 1) // f1的接收通道,f2的发送通道
    cf2 := make(chan bool, 1) // f2的接收通道,f1的发送通道

    // 初始时,将一个令牌放入cf1,让f1能够首先启动其临界区
    cf1 <- true

    // 启动两个Goroutine
    go f1(cf1, cf2, 1) // f1 接收cf1的令牌,完成后将令牌发送到cf2
    go f2(cf2, cf1, 2) // f2 接收cf2的令牌,完成后将令牌发送到cf1

    // 为了防止main Goroutine过早退出,导致子Goroutine无法完成,
    // 这里使用select{}来阻塞main Goroutine。
    // 在实际生产环境中,更推荐使用sync.WaitGroup来精确等待所有Goroutine完成。
    // 例如:
    // var wg sync.WaitGroup
    // wg.Add(2) // 假设f1和f2内部有wg.Done()
    // go f1(cf1, cf2, 1, &wg)
    // go f2(cf2, cf1, 2, &wg)
    // wg.Wait()
    select {}
}

运行上述代码,你将看到CS1和CS2的执行日志严格交替出现,证明了双通道模式的有效性。

模式的优势与扩展

  • 严格交替: 该模式确保了临界区按照预设的顺序严格交替执行,不会出现乱序或并发执行的情况。
  • 简洁明了: 通过通道的发送和接收操作,清晰地表达了令牌的传递和Goroutine的等待机制。
  • 高度可扩展: 这种模式不仅限于两个Goroutine。如果你有 f1, f2, f3 需要交替执行,你可以创建 cf1, cf2, cf3 三个通道,让 f1 将令牌传给 f2,f2 传给 f3,f3 再传回 f1,形成一个环形链。

关键考量与最佳实践

  1. 通道缓冲大小: 必须使用缓冲大小为1的通道。如果通道是无缓冲的,发送操作将阻塞直到有接收者准备好,这在某些情况下可能导致死锁或逻辑复杂化。缓冲为1确保了令牌的唯一性,即每次只有一个Goroutine持有执行权。
  2. 初始令牌: 必须在程序开始时向第一个Goroutine的接收通道发送一个令牌,否则所有Goroutine都将阻塞等待令牌,导致死锁。
  3. 主Goroutine的等待: main 函数必须等待所有子Goroutine完成其工作,否则程序可能会在子Goroutine完成前退出。在示例中使用了 select {} 来无限期阻塞 main,但这通常不是生产环境的最佳实践。更推荐使用 sync.WaitGroup 来精确管理Goroutine的生命周期。
  4. 错误处理: 在实际应用中,需要考虑Goroutine异常退出或通道关闭的情况,这可能导致令牌传递中断,从而引发死锁。

总结

双通道模式为Go语言中实现并发临界区严格交替执行提供了一个优雅且高效的解决方案。通过构建一个令牌传递的闭环,它能够精确控制Goroutine的执行顺序,并具备良好的可扩展性,适用于需要精细协调并发行为的场景。理解并正确运用这一模式,将有助于编写出更加健壮和可控的Go并发程序。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

234

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

450

2023.09.25

go语言编程软件有哪些
go语言编程软件有哪些

go语言编程软件有Go编译器、Go开发环境、Go包管理器、Go测试框架、Go文档生成器、Go代码质量工具和Go性能分析工具等。本专题为大家提供go语言相关的文章、下载、课程内容,供大家免费下载体验。

254

2023.10.13

0基础如何学go语言
0基础如何学go语言

0基础学习Go语言需要分阶段进行,从基础知识到实践项目,逐步深入。php中文网给大家带来了go语言相关的教程以及文章,欢迎大家前来学习。

701

2023.10.26

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

194

2024.02.23

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

233

2024.02.23

go语言开发工具大全
go语言开发工具大全

本专题整合了go语言开发工具大全,想了解更多相关详细内容,请阅读下面的文章。

284

2025.06.11

go语言引用传递
go语言引用传递

本专题整合了go语言引用传递机制,想了解更多相关内容,请阅读专题下面的文章。

161

2025.06.26

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

热门下载

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

精品课程

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

共32课时 | 4.4万人学习

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号