当前位置: 首页 > news >正文

golang学习笔记21——golang协程管理及sync.WaitGroup的使用

  • 推荐学习文档
    • golang应用级os框架,欢迎star
    • golang应用级os框架使用案例,欢迎star
    • 案例:基于golang开发的一款超有个性的旅游计划app经历
    • golang实战大纲
    • golang优秀开发常用开源库汇总
    • 想学习更多golang知识,这里有免费的golang学习笔记专栏

文章目录

    • 引言
    • 协程管理中的常见问题
      • 1.协程泄漏
      • 2.协程过多导致资源耗尽
    • 解决方案
      • 1.避免协程泄漏
      • 2.限制协程数量
    • 总结

引言

Golang 中的协程(goroutine)为并发编程带来了极大的便利,但在实际开发中,如果对协程管理不当,也会产生一系列问题。本文将深入探讨这些问题,并结合代码示例给出相应的解决方案。

协程管理中的常见问题

1.协程泄漏

  • 协程在执行过程中,如果由于某些原因(如阻塞在某个通道上、陷入死锁等)没有正常退出,就会导致协程泄漏。大量的协程泄漏会耗尽系统资源,如内存等。
  • 示例代码:
package mainimport ("fmt""time"
)func leakyGoroutine() {// 这个协程会一直阻塞,导致协程泄漏<-make(chan int)
}func main() {for i := 0; i < 10; i++ {go leakyGoroutine()}// 主线程休眠一段时间,让协程有机会执行time.Sleep(5 * time.Second)fmt.Println("程序结束,但协程泄漏了")
}

2.协程过多导致资源耗尽

  • 创建过多的协程而没有进行有效的限制和管理,会使系统资源(如 CPU 时间片、内存等)被大量占用,从而影响系统的性能和稳定性。
  • 示例代码:
package mainimport ("fmt""runtime""sync"
)func manyGoroutines() {var wg sync.WaitGroupfor i := 0; i < 100000; i++ {wg.Add(1)go func() {// 模拟协程执行一些简单的操作for j := 0; j < 1000; j++ {_ = j}wg.Done()}()}wg.Wait()
}func main() {before := runtime.NumGoroutine()manyGoroutines()after := runtime.NumGoroutine()fmt.Printf("创建前协程数量: %d, 创建后协程数量: %d\n", before, after)
}

解决方案

1.避免协程泄漏

  • 合理使用通道和超时机制
    • 对于可能阻塞的通道操作,可以设置超时时间,避免协程无限制地等待。
  • 代码示例:
package mainimport ("fmt""time"
)func nonLeakyGoroutine() {// 创建一个带超时的通道timeout := time.After(3 * time.Second)ch := make(chan int)go func() {// 模拟可能阻塞的操作time.Sleep(5 * time.Second)ch <- 1}()select {case <-ch:fmt.Println("协程正常接收数据")case <-timeout:fmt.Println("操作超时,协程退出")}
}func main() {for i := 0; i < 10; i++ {go nonLeakyGoroutine()}// 主线程休眠一段时间time.Sleep(5 * time.Second)fmt.Println("程序结束,没有协程泄漏")
}
  • 避免死锁
    • 在多个协程之间进行同步和通信时,要确保资源的获取和释放顺序正确,避免出现死锁导致协程无法退出。

2.限制协程数量

  • 使用信号量(Semaphore)
    • 通过信号量来限制同时执行的协程数量。
  • 代码示例:
package mainimport ("fmt""sync"
)// 定义信号量
var semaphore = make(chan struct{}, 10)func limitedGoroutine() {// 获取信号量semaphore <- struct{}{}defer func() {// 释放信号量<-semaphore}()// 协程执行的操作fmt.Println("协程执行中...")
}func main() {var wg sync.WaitGroupfor i := 0; i < 100; i++ {wg.Add(1)go func() {limitedGoroutine()wg.Done()}()}wg.Wait()fmt.Println("所有协程执行完毕")
}

总结

在 Go 语言中,协程管理是并发编程的关键部分。通过避免协程泄漏和合理限制协程数量等措施,可以有效地提高程序的性能和稳定性,充分发挥 Go 语言在并发编程方面的优势。

关注我看更多有意思的文章哦!👉👉

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • leetcode 199.二叉树的右视图
  • docker容器中的内存占用高的问题分析
  • MybatisPlus实现多租户 全局拦截器
  • 学习图解算法 使用C语言
  • MySQL基于GTID同步模式搭建主从复制
  • QUIC的丢包处理
  • 论文阅读笔记 --- 图模互补:知识图谱与大模型融合综述 --- 按参考文献整理
  • 操作系统名词_文件下载_反弹shell_1
  • C#/.NET/.NET Core技术前沿周刊 | 第 5 期(2024年9.9-9.15)
  • spring security OAuth2 搭建资源服务器以及授权服务器/jdbc/jwt两种方案
  • 【Spark性能调优】长尾任务如何优化
  • robosuite基础教程(一)——基本概念
  • 如何解决“json schema validation error ”错误? -- HarmonyOS自学6
  • fiddler抓包01:工具介绍
  • 时序必读论文11|ICLR23 TimesNet时间序列分析的二维变化建模
  • angular2开源库收集
  • C学习-枚举(九)
  • eclipse(luna)创建web工程
  • es6要点
  • SpringBoot 实战 (三) | 配置文件详解
  • supervisor 永不挂掉的进程 安装以及使用
  • Vue2 SSR 的优化之旅
  • 对话 CTO〡听神策数据 CTO 曹犟描绘数据分析行业的无限可能
  • 关于extract.autodesk.io的一些说明
  • 京东美团研发面经
  • 猫头鹰的深夜翻译:Java 2D Graphics, 简单的仿射变换
  • 什么是Javascript函数节流?
  • 问题之ssh中Host key verification failed的解决
  • 验证码识别技术——15分钟带你突破各种复杂不定长验证码
  • d²y/dx²; 偏导数问题 请问f1 f2是什么意思
  • 组复制官方翻译九、Group Replication Technical Details
  • ​直流电和交流电有什么区别为什么这个时候又要变成直流电呢?交流转换到直流(整流器)直流变交流(逆变器)​
  • # centos7下FFmpeg环境部署记录
  • #include<初见C语言之指针(5)>
  • (1/2)敏捷实践指南 Agile Practice Guide ([美] Project Management institute 著)
  • (173)FPGA约束:单周期时序分析或默认时序分析
  • (C语言)fgets与fputs函数详解
  • (附源码)springboot家庭财务分析系统 毕业设计641323
  • (附源码)springboot优课在线教学系统 毕业设计 081251
  • (黑马C++)L06 重载与继承
  • (论文阅读22/100)Learning a Deep Compact Image Representation for Visual Tracking
  • (每日一问)设计模式:设计模式的原则与分类——如何提升代码质量?
  • (算法)大数的进制转换
  • (一)硬件制作--从零开始自制linux掌上电脑(F1C200S) <嵌入式项目>
  • (原創) 人會胖會瘦,都是自我要求的結果 (日記)
  • (转)ObjectiveC 深浅拷贝学习
  • ****** 二 ******、软设笔记【数据结构】-KMP算法、树、二叉树
  • .NET / MSBuild 扩展编译时什么时候用 BeforeTargets / AfterTargets 什么时候用 DependsOnTargets?
  • .NET CLR基本术语
  • .Net Core webapi RestFul 统一接口数据返回格式
  • .net core webapi 部署iis_一键部署VS插件:让.NET开发者更幸福
  • .net php 通信,flash与asp/php/asp.net通信的方法
  • .Net--CLS,CTS,CLI,BCL,FCL
  • :O)修改linux硬件时间
  • @data注解_SpringBoot 使用WebSocket打造在线聊天室(基于注解)