0

0

如何在Go语言中限制协程数量并避免死锁问题?

碧海醫心

碧海醫心

发布时间:2025-03-20 11:04:31

|

731人浏览过

|

来源于php中文网

原创

go语言协程数量限制与死锁避免:高效处理并发任务

在Go语言并发编程中,限制协程数量以避免资源耗尽至关重要。然而,不当的协程控制和数据传递方式可能导致死锁。本文将分析一个使用sync.WaitGroup和通道进行协程数量限制和数据传递的案例,并提出解决方案,避免fatal error: all goroutines are asleep - deadlock!错误。

如何在Go语言中限制协程数量并避免死锁问题?

问题描述

示例代码使用一个通道c限制并发协程数,另一个通道creport用于收集协程结果。由于不恰当的通道关闭和数据读取方式,导致死锁。

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

问题代码分析与改进

原始代码中存在两个主要问题:

  1. defer close(creport) 的错误放置: defer close(creport)语句在主函数中,但由于主循环持续读取creport通道,该语句永远无法执行,导致creport通道无法关闭,从而阻塞协程。

    盛世企业网站管理系统1.1.2
    盛世企业网站管理系统1.1.2

    免费 盛世企业网站管理系统(SnSee)系统完全免费使用,无任何功能模块使用限制,在使用过程中如遇到相关问题可以去官方论坛参与讨论。开源 系统Web代码完全开源,在您使用过程中可以根据自已实际情况加以调整或修改,完全可以满足您的需求。强大且灵活 独创的多语言功能,可以直接在后台自由设定语言版本,其语言版本不限数量,可根据自已需要进行任意设置;系统各模块可在后台自由设置及开启;强大且适用的后台管理支

    下载
  2. 重复的读取操作: 主函数和子协程都试图从creport通道读取数据,造成资源竞争和死锁风险。

为了解决这些问题,我们进行如下修改:

  • 移除defer close(creport):creport通道的关闭操作移至wg.Wait()之后,确保所有协程都已完成任务。

  • 优化主函数数据接收: 移除主函数中重复的creport通道读取操作,改为在wg.Wait()之后处理结果。

  • 清晰的错误处理:select语句中添加ok判断,处理通道关闭情况,避免panic。

改进后的代码 (示例,需根据实际代码调整):

package main

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

// ... (省略部分代码,假设Bag类型已定义)

func main() {
    // ... (省略部分代码)

    var (
        cLimit  = 3 // 协程限制数量
        wg      = sync.WaitGroup{}
    )

    c := make(chan int, cLimit)
    c2 := make(chan int, 10)
    cReport := make(chan Bag, 100)
    defer close(c) // 保持原有c通道的关闭

    // 启动工作协程
    for i := 0; i < 10; i++ {
        wg.Add(1)
        go func(i int) {
            defer wg.Done()
            c <- 1 // 获取执行许可
            defer func() { <-c }() // 释放执行许可
            // ... (协程工作逻辑,并将结果发送到cReport)
            cReport <- Bag{data: fmt.Sprintf("Result %d", i)}
        }(i)
    }

    wg.Wait() // 等待所有协程完成
    close(cReport) // 在所有协程结束后关闭cReport通道

    // 处理结果
    for result := range cReport {
        fmt.Println("Received:", result.data)
    }

    log.Println("All goroutines finished.")
}

通过以上改进,我们有效地避免了死锁,并确保了协程的正确管理和数据处理。 记住,在并发编程中,仔细考虑通道的关闭时机和避免资源竞争至关重要。 根据实际业务逻辑调整代码,确保所有通道在所有协程结束后再关闭。

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

265

2023.10.25

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语言相关的文章、下载、课程内容,供大家免费下载体验。

246

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、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

225

2024.02.23

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

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

74

2025.12.31

热门下载

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

精品课程

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

共28课时 | 4万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 2.2万人学习

Go 教程
Go 教程

共32课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号