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

golang goroutine(协程)和 channel(管道) 案例解析

文章目录

    • goroutine和channel概念
    • 开启线程与channel简单通信流程
    • 多个工作协程并发执行流程

goroutine和channel概念

  • goroutine(协程),一般我们常见的是进程,线程,进程可以理解为一个软件在运行执行的过程,线程跟协程比较类似,都是单独开辟一块内存,异步执行。不同的是协程占的内存比较小初始栈2KB左右。
  • channel(管道) , 管道实则也是用来存储数据的,那为什么不用切片呢,channel主要用于多个协程之间的通信同步,遵循的是先进先出,channel有容量的限制,如果说一直往里面存,不取的话那管道就会阻塞报错,所以合理处理管道数据,当数据处理完成后,要将管道关闭,不然后面如果要遍历channel时,就会一直等待channel关闭才会执行完成。

开启线程与channel简单通信流程

这里是使用了sync.awitGroup实现多协程异步执行,与channel进行通信过程。


//使用awitGroup用于阻塞主进程作用 等待所有协程执行完成,解除阻塞(类似于计数器,计数器为0,解除阻塞)
var wg sync.awitGroupfunc printNum (ch chan int){for i :=1; i < 10 i++{//循环将i添加到ch管道中,(channel为引用类型,可以直接修改源数据)ch <- i}//关闭管道 在后面我们如果要遍历处理管道数据时,不关闭则接收方会无线阻塞close(ch)//计数器减1wg.Done()
}func readNum (ch chan int){//往ch管道中取数据 打印出来 ,channel时数据消耗性,只要读取了,这条数据不会在channel中保留、for v := range ch{fmt,printIn(v)}}func main {//创建一个channel管道,int类型,容量为10(缓冲区为10)ch:= make(chan int ,10)//计数器加1wg.add(1)//go关键字 开启一个协程 go printNum(ch)//计数器加1wg.add(1)//开启一个协程  协程与协程go readNum(ch)//阻塞主进程wg.awit()fmt.printIn("执行完成")
}

多个工作协程并发执行流程

处理100万个数字内那些是素数,这种数量大的情况下,循环创建多个工作协程,同时执行相关逻辑。

//使用awitGroup用于阻塞主进程作用 等待所有协程执行完成,解除阻塞(类似于计数器,计数器为0,解除阻塞)
var wg sync.awitGroupfunc generateNumbers(inputChan chan int){for i :=2; i < 1000000 i++{//循环将i添加到ch管道中,(channel为引用类型,可以直接修改源数据)inputChan  <- i}//关闭管道 在后面我们如果要遍历处理管道数据时,不关闭则接收方会无线阻塞close(ch)//计数器减1wg.Done()
}func worker(inputChan chan int , outputChan chan int,exitChan chan int){//inputChan 管道中取数据 打印出来 ,channel是数据消耗性,只要读取了,这条数据不会在channel中保留、所以循环worker线程是随机值,是不可能多个工作线程执行一个数值for num := range inputChan {flag := truefor i := 2; i < num; i++ {// 判断如果不为素数 flag为false 则break跳出循环if num%i == 0 {flag = falsebreak}}if flag {primeChan <- num //num 为素数 放在outputChan 管道中}}// 标记减一  与wg.add(1) 配套使用wg.Done()// 因为worker需要多次执行 所以不能直接关闭outputChan 管道// 向exitChan管道中写入数据 记录退出信号exitChan <- true}func listenWorker(exitChan chan int ,outputChan chan int){for i:=1; i < 16; i++{/*从oututChan中取数据,如果循环过程中执行的比工作协程快,没有取到数据,则会阻塞等待,直到outputChan中拿到数据,循环16次,全部协程执行完成。*/<-outputChan}//关闭outputChan管道close(outputChan)//结束监听携程wg.Done()
}func main {//存储100万个待处理数字inputChan:= make(chan int ,1000)//存素数的channel管道outputChan:= make(chan int 1000)//监听多协程结束的信号 容量为16,因为我们就开了16个协程,一般为cpu的逻辑处理器数量exitChan := make(chat bool 16)//计数器加1wg.add(1)//go关键字 开启一个协程 go generateNumbers(inputChan)// 创建工作协程numWorkers := runtime.NumCPU() // 协程数 = CPU 核心数for i := 0; i < numWorkers; i++ {go worker(inputChan, outputChan)}//如何关闭循环的多协程,第二种方案 再开一协程用于实时监听工作协程工作状态wg.add(1)go listenWorker(exitChan,outputChan)//阻塞主进程wg.awit()//如何关闭循环的多协程,第一种方案 在所有协程完成之后,关闭存素数的管道(outputChan)close(outputChan)fmt.printIn("执行完成")
}
http://www.xdnf.cn/news/169453.html

相关文章:

  • 【Java】jdk动态代理
  • Flink02-学习-套接字分词
  • Web前端开发:CSS Float(浮动)与 Positioning(定位)
  • 数据结构——二叉树和堆(万字,最详细)
  • 【AI论文】RefVNLI:迈向可扩展的主题驱动文本到图像生成评估
  • SLAM技术:从原理到应用的全面解析
  • 计算机网络 | 应用层(6) -- 套接字编程
  • Java自定义注解详解
  • 「Mac畅玩AIGC与多模态01」架构篇01 - 展示层到硬件层的架构总览
  • 深度学习常见框架:TensorFlow 与 PyTorch 简介与对比
  • 在 Ubuntu 24.04 系统上安装和管理 Nginx
  • body Param Query 三个 不同的入参 分别是什么意思 在前端 要怎么传 这三种不同的参数
  • DAY7-C++进阶学习
  • Python爬虫课程实验指导书
  • 麒麟系统搭建离线NTP服务器,让局域网内windows系统同步其时间,并付排错避坑思路
  • Android Studio中创建第一个Flutter项目
  • 前端性能优化面试回答技巧
  • django admin 去掉新增 删除
  • 【愚公系列】《Manus极简入门》005-DeepSeek与Manus的创新之处
  • PostSwigger Web 安全学习:CSRF漏洞3
  • C# 利用log4net 工作台打印和保存到文件
  • 央视两次采访报道爱藏评级,聚焦生肖钞市场升温,评级币成交易安全“定心丸”
  • C# 类的基本概念(类成员)
  • 16bit 高精度低延时霍尔角度编码器,KTH7824,替代MA730
  • 高自由度与多功能指尖设计:Allegro灵巧手V5(4F Plus)的技术亮点
  • 电商数据采集电商,行业数据分析,平台数据获取|稳定的API接口数据
  • 榕壹云国际版短剧系统:基于Spring Boot+MySQL+UniApp的全球短剧创作平台
  • Scala 函数柯里化及闭包
  • 用 Nodemon 解决 npm run serve 频繁重启服务
  • 2个小时1.5w字| React Golang 全栈微服务实战