0

0

Go并发:理解select{}的非阻塞行为与避免死锁的策略

心靈之曲

心靈之曲

发布时间:2025-09-01 11:16:01

|

788人浏览过

|

来源于php中文网

原创

Go并发:理解select{}的非阻塞行为与避免死锁的策略

本文深入探讨了Go语言中select{}语句的特殊行为,解释了为何它在没有case时无法实现永久阻塞以等待其他Goroutine完成,并可能导致死锁。文章提供了两种避免此类并发死锁的有效策略:利用sync.WaitGroup精确等待所有Goroutine完成,以及构建灵活的并发工作池模型来管理任务和结果,从而确保并发程序的健壮性和正确性。

select{}的非阻塞特性与死锁分析

go语言中,select{}语句用于在多个通信操作中选择一个就绪的执行。当select语句不包含任何case分支时,其行为是特殊的:它不会无限期地等待其他goroutine完成。相反,如果当前goroutine是唯一一个可运行的goroutine,且select{}没有任何可执行的case(因为根本没有case),go运行时会立即检测到所有goroutine都已休眠,从而抛出“all goroutines are asleep - deadlock!”的错误。

考虑以下代码示例,它尝试使用通道作为信号量来限制并发Goroutine的数量:

package main

import (
    "fmt"
    "math/rand"
    "time"
)

func runTask(t string, ch *chan bool) {
    start := time.Now()
    fmt.Println("starting task", t)
    time.Sleep(time.Millisecond * time.Duration(rand.Int31n(1500))) // 模拟处理时间
    fmt.Println("done running task", t, "in", time.Since(start))
    <-*ch // 任务完成后,释放一个信号量
}

func main() {
    numWorkers := 3
    files := []string{"a", "b", "c", "d", "e", "f", "g", "h", "i", "j"}

    activeWorkers := make(chan bool, numWorkers) // 容量为numWorkers的缓冲通道作为信号量

    for _, f := range files {
        activeWorkers <- true // 尝试获取一个信号量,如果通道已满则阻塞
        fmt.Printf("activeWorkers is %d long.\n", len(activeWorkers))
        go runTask(f, &activeWorkers)
    }
    select{} // 期望阻塞主Goroutine,等待所有任务完成
}

上述代码会触发死锁。其原因在于:

  1. 主Goroutine过早阻塞: main Goroutine在启动所有runTask Goroutine后,立即执行select{}。由于select{}没有case,它会检查是否有其他Goroutine可运行。
  2. runTask Goroutine的阻塞: runTask Goroutine在完成模拟处理后,会尝试从activeWorkers通道中接收一个值(
  3. 死锁发生: 当main Goroutine执行到select{}时,它自身进入休眠状态。此时,runTask Goroutines可能仍在执行或已完成并尝试释放信号量。由于main Goroutine已休眠,它不会再向activeWorkers通道发送任何值,也不会从其中接收值。最终,所有runTask Goroutine都会尝试从activeWorkers通道接收,但通道中没有可接收的值(因为它们之前已经发送了true,现在需要接收来释放),它们将全部阻塞。当Go运行时发现所有Goroutine都处于阻塞状态,且没有 Goroutine能够唤醒其他 Goroutine时,便会判定为死锁。select{}在此场景下,并没有起到等待其他Goroutine的作用,反而成为了死锁的直接原因。

避免死锁的策略

为了正确地等待所有并发任务完成并避免死锁,Go语言提供了更合适的同步原语。

1. 使用sync.WaitGroup等待Goroutine完成

sync.WaitGroup是Go标准库中用于等待一组Goroutine完成的机制。它通过一个计数器来实现:调用Add(n)增加计数器,每个Goroutine完成时调用Done()减少计数器,最后通过Wait()阻塞直到计数器归零。

结合信号量通道,我们可以这样修改上述代码:

package main

import (
    "fmt"
    "math/rand"
    "sync" // 引入sync包
    "time"
)

func runTaskWithWaitGroup(t string, ch chan bool, wg *sync.WaitGroup) {
    defer wg.Done() // 确保Goroutine完成时调用Done()
    start := time.Now()
    fmt.Println("starting task", t)
    time.Sleep(time.Millisecond * time.Duration(rand.Int31n(1500))) // 模拟处理时间
    fmt.Println("done running task", t, "in", time.Since(start))
    <-ch // 任务完成后,释放一个信号量
}

func main() {
    numWorkers := 3
    files := []string{"a", "b", "c", "d", "e", "f", "g", "h", "i", "j"}

    activeWorkers := make(chan bool, numWorkers) // 缓冲通道作为信号量
    var wg sync.WaitGroup                        // 声明WaitGroup

    for _, f := range files {
        activeWorkers <- true // 获取一个信号量,控制并发数
        wg.Add(1)             // 每启动一个Goroutine,计数器加1
        fmt.Printf("activeWorkers is %d long.\n", len(activeWorkers))
        go runTaskWithWaitGroup(f, activeWorkers, &wg)
    }

    wg.Wait() // 阻塞主Goroutine,直到所有Goroutine都调用Done()
    fmt.Println("所有任务已完成。")
}

在这个修正后的版本中:

Moshi Chat
Moshi Chat

法国AI实验室Kyutai推出的端到端实时多模态AI语音模型,具备听、说、看的能力,不仅可以实时收听,还能进行自然对话。

下载
  • main Goroutine在每次启动runTaskWithWaitGroup时调用wg.Add(1)。
  • runTaskWithWaitGroup Goroutine在执行结束前(通过defer wg.Done())调用wg.Done()。
  • main Goroutine在循环结束后调用wg.Wait(),这将阻塞main Goroutine,直到所有runTaskWithWaitGroup Goroutine都完成并调用了Done(),从而避免了死锁。

2. 构建并发工作池模型

对于更复杂的任务调度和结果收集场景,Go语言中更推荐使用工作池(Worker Pool)模式。这种模式通常涉及一个输入通道用于分发任务,一组固定的工作Goroutine处理任务,以及一个可选的输出通道用于收集结果。

以下是一个经典的并发工作池实现:

package main

import (
    "fmt"
    "math/rand"
    "time"
)

// runTask 模拟任务执行,返回任务名称
func runTask(t string) string {
    start := time.Now()
    fmt.Println("starting task", t)
    time.Sleep(time.Millisecond * time.Duration(rand.Int31n(1500))) // 模拟处理时间
    fmt.Println("done running task", t, "in", time.Since(start))
    return t // 返回任务结果
}

// worker Goroutine从in通道接收任务,处理后将结果发送到out通道
func worker(in chan string, out chan string) {
    for t := range in { // 循环从in通道接收任务,直到in通道关闭
        out <- runTask(t) // 处理任务并将结果发送到out通道
    }
}

func main() {
    numWorkers := 3
    files := []string{"a", "b", "c", "d", "e", "f", "g", "h", "i", "j"}

    // 创建输入和输出通道
    in, out := make(chan string), make(chan string)

    // 启动固定数量的工作Goroutine
    for i := 0; i < numWorkers; i++ {
        go worker(in, out)
    }

    // 启动一个Goroutine来调度任务(将任务发送到in通道)
    go func() {
        for _, f := range files {
            in <- f // 发送任务
        }
        close(in) // 所有任务发送完毕后,关闭in通道,通知worker Goroutine停止接收
    }()

    // 主Goroutine从out通道接收所有任务的结果
    // 循环次数等于任务总数,确保接收所有结果
    for i := 0; i < len(files); i++ {
        <-out // 接收结果
    }
    // 此时,所有任务都已处理并接收了结果,可以安全退出
    fmt.Println("所有任务已完成并收集到结果。")
}

这种工作池模式的优势在于:

  • 明确的并发控制: 通过启动固定数量的worker Goroutine来限制并发数。
  • 任务解耦: 任务的提交和处理逻辑分离,更易于管理。
  • 结果收集: out通道可以方便地收集所有任务的处理结果。
  • 优雅关闭: 通过关闭in通道,可以通知worker Goroutine停止工作,避免资源泄露。

注意事项: 在使用range从通道读取数据时,务必在所有数据发送完毕后关闭通道(如close(in)),否则range循环会一直阻塞,导致潜在的死锁。

总结与最佳实践

  • select{}的用途: select{}(无case)并非用于等待所有Goroutine完成,它只会立即检查当前Goroutine是否可以休眠,如果其他Goroutine也最终全部阻塞,就会导致死锁。它通常用于在多个通道操作之间进行选择,或配合default分支实现非阻塞操作。
  • 等待Goroutine完成: 当需要等待一组Goroutine执行完毕时,sync.WaitGroup是首选且最简洁的同步原语。
  • 并发任务管理: 对于需要控制并发度、调度任务和收集结果的场景,并发工作池模式是一种更强大、更灵活的解决方案。它通过通道实现任务的输入、结果的输出以及工作Goroutine之间的协调。
  • 通道管理: 使用通道时,正确地关闭通道(当所有发送者完成发送后)对于使用range接收数据至关重要,它可以避免接收者无限期阻塞。

理解这些Go并发原语的正确用法,对于编写健壮、高效且无死锁的并发程序至关重要。

相关专题

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

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

233

2023.09.06

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

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

442

2023.09.25

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

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

245

2023.10.13

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

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

691

2023.10.26

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

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

187

2024.02.23

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

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

223

2024.02.23

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

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

277

2025.06.11

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

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

156

2025.06.26

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

热门下载

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

精品课程

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

共32课时 | 3.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号