
本文深入探讨Go并发编程中常见的“all goroutines are asleep - deadlock!”错误,尤其是在构建工作者系统时因未正确关闭输出通道导致的死锁。通过分析问题根源,文章将演示如何利用控制通道或sync.WaitGroup机制,实现对工作协程的有效协调,确保所有任务完成后安全关闭通道,从而优雅地终止程序,避免死锁。
在Go语言的并发编程模型中,goroutine和channel是核心构建块。然而,不当的通道使用方式,特别是通道的关闭机制,很容易导致程序进入“死锁”状态,并抛出fatal error: all goroutines are asleep - deadlock!。这个错误表明Go运行时检测到程序中所有goroutine都处于阻塞状态,且没有可以被调度的goroutine来解除这些阻塞,因此程序无法继续执行。
在一个典型的生产者-消费者或工作者池(Worker Pool)模式中,如果一个或多个goroutine正在尝试从一个通道接收数据,而这个通道的发送方已经完成其所有工作,但却忘记关闭通道,那么这些接收goroutine将永远等待下去,从而导致整个程序死锁。
考虑一个Go语言实现的工作者系统骨架,其设计目标是创建一批工作协程处理任务,并通过通道进行协调。
原始代码结构如下:
package main
import (
"bufio"
"flag"
"fmt"
"log"
"math/rand"
"os"
"time"
)
type Work struct {
id int
ts time.Duration
}
const (
NumWorkers = 5000
NumJobs = 100000
)
func worker(in <-chan *Work, out chan<- *Work) {
for w := range in {
st := time.Now()
time.Sleep(time.Duration(rand.Int63n(int64(200 * time.Millisecond))))
w.ts = time.Since(st)
out <- w
}
}
func main() {
wait := flag.Bool("w", false, "wait for <enter> before starting")
flag.Parse()
if *wait {
fmt.Printf("I'm <%d>, press <enter> to continue", os.Getpid())
reader := bufio.NewReader(os.Stdin)
reader.ReadString('\n')
}
Run()
}
func Run() {
in, out := make(chan *Work, 100), make(chan *Work, 100)
for i := 0; i < NumWorkers; i++ {
go worker(in, out)
}
go createJobs(in)
receiveResults(out)
}
func createJobs(queue chan<- *Work) {
for i := 0; i < NumJobs; i++ {
work := &Work{i, 0}
queue <- work
}
close(queue) // 输入通道在所有任务创建后关闭
}
func receiveResults(completed <-chan *Work) {
for w := range completed { // 从完成通道接收结果
log.Printf("job %d completed in %s", w.id, w.ts)
}
}在这个示例中,createJobs协程负责向in通道发送任务,并在所有任务发送完毕后正确地关闭了in通道。worker协程从in通道接收任务,处理后将结果发送到out通道。receiveResults函数则通过for w := range completed循环从out通道(在此函数中命名为completed)接收所有完成的任务结果。
死锁的根源在于: 当createJobs协程完成并关闭in通道后,所有的worker协程会逐一处理完in通道中剩余的任务,然后它们从for w := range in循环中退出。这些worker协程退出后,out通道将不再有发送者。然而,receiveResults函数中的for w := range completed循环会持续尝试从out通道接收数据。由于out通道从未被关闭,receiveResults协程将永远阻塞等待新的数据。此时,所有worker协程已退出,createJobs协程也已完成,只剩下receiveResults协程一个活跃的goroutine在无限等待一个永远不会关闭的通道,最终导致死锁。
为了解决这个问题,我们需要在所有工作协程完成其工作后,显式地关闭out通道。一种方法是引入一个额外的“控制通道”来协调工作协程的完成状态。
小云雀
剪映出品的AI视频和图片创作助手
1949
查看详情
实现步骤:
package main
import (
"bufio"
"flag"
"fmt"
"log"
"math/rand"
"os"
"time"
)
type Work struct {
id int
ts time.Duration
}
const (
NumWorkers = 5000
NumJobs = 100000
)
// worker 函数现在接收一个额外的控制通道参数
func worker(ctrl chan<- bool, in <-chan *Work, out chan<- *Work) {
defer func() {
ctrl <- true // worker 完成其所有工作后,向控制通道发送完成信号
}()
for w := range in {
st := time.Now()
time.Sleep(time.Duration(rand.Int63n(int64(200 * time.Millisecond))))
w.ts = time.Since(st)
out <- w
}
}
// control 协程负责等待所有worker完成,然后关闭输出通道
func control(ctrl <-chan bool, numWorkers int, out chan<- *Work) {
for i := 0; i < numWorkers; i++ {
<-ctrl // 等待每个worker的完成信号
}
close(out) // 所有worker完成后,关闭输出通道
}
func main() {
wait := flag.Bool("w", false, "wait for <enter> before starting")
flag.Parse()
if *wait {
fmt.Printf("I'm <%d>, press <enter> to continue", os.Getpid())
reader := bufio.NewReader(os.Stdin)
reader.ReadString('\n')
}
Run()
}
func Run() {
in, out := make(chan *Work, 100), make(chan *Work, 100)
ctrl := make(chan bool, NumWorkers) // 创建控制通道,缓冲大小为worker数量
// 启动工作协程
for i := 0; i < NumWorkers; i++ {
go worker(ctrl, in, out)
}
// 启动任务创建协程
go createJobs(in)
// 启动控制协程,它将等待所有worker完成并关闭 'out' 通道
go control(ctrl, NumWorkers, out)
// 接收结果
receiveResults(out)
}
func createJobs(queue chan<- *Work) {
for i := 0; i < NumJobs; i++ {
work := &Work{i, 0}
queue <- work
}
close(queue) // 创建任务完成后关闭输入通道
}
func receiveResults(completed <-chan *Work) {
for w := range completed {
log.Printf("job %d completed in %s", w.id, w.ts)
}
}sync.WaitGroup 是Go标准库提供的一种更通用的同步原语,用于等待一组goroutine完成。它通常比手动管理控制通道更简洁和惯用。
实现步骤:
package main
import (
"bufio"
"flag"
"fmt"
"log"
"math/rand"
"os"
"sync" // 引入 sync 包
"time"
)
type Work struct {
id int
ts time.Duration
}
const (
NumWorkers = 5000
NumJobs = 100000
)
// worker 函数现在接收一个 WaitGroup 指针
func worker(wg *sync.WaitGroup, in <-chan *Work, out chan<- *Work) {
defer wg.Done() // 确保worker退出时通知WaitGroup
for w := range in {
st := time.Now()
time.Sleep(time.Duration(rand.Int63n(int64(200 * time.Millisecond))))
w.ts = time.Since(st)
out <- w
}
}
func main() {
wait := flag.Bool("w", false, "wait for <enter> before starting")
flag.Parse()
if *wait {
fmt.Printf("I'm <%d>, press <enter> to continue", os.Getpid())
reader := bufio.NewReader(os.Stdin)
reader.ReadString('\n')
}
Run()
}
func Run() {
in, out := make(chan *Work, 100), make(chan *Work, 100)
var wg sync.WaitGroup // 声明 WaitGroup
// 启动工作协程
for i := 0; i < NumWorkers; i++ {
wg.Add(1) // 增加计数
go worker(&wg, in, out)
}
// 启动任务创建协程
go createJobs(in)
// 启动一个独立的协程来等待所有worker完成并关闭输出通道
go func() {
wg.Wait() // 等待所有worker完成
close(out) // 关闭输出通道
}()
// 接收结果
receiveResults(out)
}
func createJobs(queue chan<- *Work) {
for i := 0; i < NumJobs; i++ {
work := &Work{i, 0}
queue <- work
}
close(queue) // 创建任务完成后关闭输入通道
}
func receiveResults(completed <-chan *Work) {
for w := range completed {
log.Printf("job %d completed in %s", w.id, w.ts)
}
}正确管理Go通道是编写健壮并发程序的基石。以下是一些关键原则:
“all goroutines are asleep - deadlock!”错误是Go并发编程中常见的陷阱,通常源于通道的生命周期管理不当,特别是输出通道未被正确关闭。通过本文介绍的两种方法——使用控制通道或sync.WaitGroup——我们可以有效地协调goroutine的完成状态,确保在所有发送方都已完成工作后,能够及时关闭通道,从而避免死锁,并使程序优雅地退出。
在实际开发中,sync.W
aitGroup因其简洁性和通用性,常被视为处理此类同步问题的首选方案。理解并遵循通道管理的
以上就是Go 并发编程:避免 Goroutine 死锁与通道的优雅关闭的详细内容,更多请关注其它相关文章!
相关文章:
Lar*el头像管理:图片缩放与旧文件删除的最佳实践
铁路12306卧铺选择攻略 铁路12306下铺座位预定技巧
智慧团建扫码登录入口 智慧团建扫码登录入口官网版
整合Supabase认证与Django模型:跨模式迁移的解决方案
win11 arm版怎么安装 M1/M2 Mac虚拟机安装ARM win11的方法
J*aScript中高效管理与清空动态列表:避免循环陷阱
J*aScript数据结构转换:将对象数组按类别分组
MAC如何安全彻底地删除文件_MAC使用终端命令确保文件无法被恢复
Go与Ruby之间实现AES加密互通:CFB模式下的密钥长度匹配策略
PySpark中高效提取字符串右侧可变长度数字:使用regexp_extract
WooCommerce后台产品编辑页:获取分类ID并实现角色权限控制
使用 Pandas 高效处理 .dat 文件:字符清理与数据计算
汽水音乐在线解析 汽水音乐在线解析入口
Win11网速慢怎么解决 Win11网络设置优化解除限速
新三国志曹操传110级星符试炼夏侯渊极难攻略
内存检查:在VS Code中调试C++时的内存视图
win11如何卸载Windows更新补丁 Win11解决更新导致系统不稳定的问题【修复】
PS5 Pro有点优势但不多! 《燕云十六声》PS5平台与PC性能画面对比
文心一言怎样用插件调度API数据_文心一言用插件调度API数据【API调用】
Lar*el 递归关系中排除指定分支的教程
PHP表单隐藏域数据传递:常见问题与最佳实践
动漫岛观看全网网 动漫岛在线正版动漫入口
ACG动漫手机版官网入口 手机ACG动漫APP在线观看正版
J*a应用程序首次运行自动创建文件与目录的最佳实践
在Go语言中利用后缀数组处理多字符串:实现高效文本匹配与自动补全
在Runstone环境中高效处理TasteDive API的JSON数据
反效果?《战地6》免费试玩开启后玩家数不升反降
React Router 嵌套组件中 URL 重定向问题的解决方案
天眼查企业查询官网入口 天眼查官方网页版查询
mysql如何设置表访问权限_mysql表访问权限配置
PHP高效扁平化嵌套数组:使用array_merge与数组解包操作符
PySpark中从现有列右侧提取可变长度字符创建新列的教程
苹果手机如何防止被恶意App追踪
使用PHP DOM解析器高效提取HTML中特定标题及其紧邻段落
Golang如何实现简单的Web表单_Golang表单提交与验证处理方法
在命令行怎么运行html项目_命令行运行html项目方法【教程】
解决Python单元测试中Mock异常方法调用计数为零的问题
神经网络二分类模型训练异常:高损失与完美验证准确率的排查与修正
在python-socketio事件处理器中安全访问Flask应用上下文
解决Flask中Quill编辑器内容提交失败及TypeError的指南
QQ邮箱电脑版登录入口_QQ邮箱官方网站登录平台
谷歌google账号怎么注册账号 谷歌账号注册官方流程
Win11怎么开启高性能模式_Windows 11电源计划优化设置
谷歌邮箱网页版官方页面入口 谷歌邮箱网页端快速访问
Win11怎么关闭触摸屏_Windows 11禁用HID符合标准触摸屏
C++如何检测键盘输入_C++ _kbhit与_getch函数非阻塞输入
菜鸟取件码是什么怎么查 最全查询渠道汇总
Sublime怎么配置Nim语言环境_Sublime Nim代码高亮与补全
Node.js中HTML按钮与J*aScript函数交互的正确姿势
魅族17怎样用浏览器译外语网页_iPhone魅族17浏览器译外语网页【即时翻译】