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

【go语言 | 第5篇】channel——多个goroutine之间通信

文章目录

  • channel的定义和使用
  • channel——有缓冲和无缓冲同步
    • 1. 无缓冲的channel
    • 2. 有缓冲的channel
  • channel——关闭channel
  • channel 与 range
  • channel 与 select

channel的定义和使用

channel 用于多个 goroutine 之间的通信

packagemainimport"fmt"funcmain(){// 创建一个 channelc:=make(chanint)gofunc(){deferfmt.Println("goroutine结束!")fmt.Println("goroutine正在运行")// 将数据发送到 channel 中c<-100}()// 从 channel 中接收数据, 存入到 numnum:=<-c fmt.Println("num:",num)fmt.Println("main goroutine 结束")}

为什么 main goroutine 中的执行会在 子goroutine 之后?

(1)main go 已经到达 num := <- c 时,如果 channel 中还没有存在 c,就会对 main go 进行阻塞。
(2)sub go 已经到达 c <- 100,如果 main go 还没有到达 num := <- c,因为此时的管道 c 是无缓冲的,就会对 sub go 进行阻塞。

channel——有缓冲和无缓冲同步

1. 无缓冲的channel


(1)在第1步,两个goroutine都到达通道,但哪个都没有开始执行发送或者接收。
(2)在第2步,左侧的goroutine将它的手伸进了通道,这模拟了向通道发送数据的行为。这时,这个goroutine会在通道中被锁住,直到交换完成。
(3)在第3步,右侧的goroutine将它的手放入通道,这模拟了从通道里接收数据。这个goroutine一样也会在通道中被锁住,直到交换完成。
(4)在第4步和第5步,进行交换,并最终,在第6步,两个goroutine都将它们的手从通道里拿出来,这模拟了被锁住的goroutine得到释放。两个goroutine现在都可以去做其他事情了。

2. 有缓冲的channel

packagemainimport("fmt""time")funcmain(){// 创建一个容量为3的 channel(有缓冲的channel)c:=make(chanint,3)gofunc(){deferfmt.Println("子go程结束!")fori:=0;i<3;i++{c<-i fmt.Println("子go程 i = ",i," 元素个数:",len(c)," 元素个数:",cap(c))}}()time.Sleep(1*time.Second)fori:=0;i<3;i++{// 从 channel 中获取数据num:=<-c fmt.Println("num =",num)}fmt.Println("main goroutine 运行结束")}



(1)在第1步,右侧的goroutine正在从通道接收一个值。
(2)在第2步,右侧的这个goroutine独立完成了接收值的动作,而左侧的goroutine正在发送一个新值到通道里。
(3)在第3步,左侧的goroutine还在向通道发送新值,而右侧的goroutine正在从通道接收另外一个值。这个步骤里的两个操作既不是同步的,也不会互相阻塞。
(4)最后,在第4步,所有的发送和接收都完成,而通道里还有几个值,也有一些空间可以存更多的值。

channel——关闭channel

packagemainimport"fmt"funcmain(){c:=make(chanint)gofunc(){fori:=0;i<3;i++{c<-i}// close 关闭 channelclose(c)}()for{// ok 为 true 表示 channel 没有关闭, ok 为 false 表示 channel 已经关闭ifdata,ok:=<-c;ok{fmt.Println("data: ",data)}else{break}}fmt.Println("main goroutine 结束")}


(1)channel不像文件一样需要经常去关闭,只有当你确实没有任何发送数据了,或者想显式的结束range循环之类的,才去关闭channel;
(2)关闭channel后,无法向channel再发送数据(引发panic错误后导致接收立即返回零值);
(3)关闭channel后,可以继续从channel接收数据;
(4)对于nil channel,无论收发都会被阻塞。

channel 与 range

packagemainimport"fmt"funcmain(){c:=make(chanint)gofunc(){fori:=0;i<3;i++{c<-i}// close 关闭 channelclose(c)}()/* for { // ok 为 true 表示 channel 没有关闭, ok 为 false 表示 channel 已经关闭 if data, ok := <- c; ok { fmt.Println("data: ", data) } else { break } } */// 可以使用 range 代替不断迭代的操作的cahnnelfordata:=rangec{fmt.Println("data: ",data)}fmt.Println("main goroutine 结束")}

channel 与 select

单流程下,一个go只能监控一个 channel 状态,select可以完成监控多个 channel的状态。

packagemainimport("fmt")funcmethod(c,quitchanint){x,y:=1,1for{select{casec<-x:// 如果 c 可写入,执行下面x=y y=x+ycase<-quit:fmt.Println("quit.....")return}}}funcmain(){c:=make(chanint)quit:=make(chanint)gofunc(){fori:=0;i<6;i++{fmt.Println(<-c)}quit<-0}()method(c,quit)}

http://www.jsqmd.com/news/101879/

相关文章:

  • 程序员职场生存图鉴,真防御性编程
  • 开源医疗相关数据集
  • LobeChat公益活动策划方案生成
  • LobeChat品牌形象重塑方案生成
  • 人脸识别项目如何在Spring Boot项目中如何建立数据库和管理
  • 主存编址是什么
  • Spring Boot应用开发全攻略:从零基础到企业级实战
  • 运维系列数据库系列【仅供参考】:达梦数据库还原之指定映射路径还原
  • LobeChat老带新奖励机制设计
  • 运维系列数据库系列【仅供参考】:达梦逻辑导入使用总结
  • 当AI面临伦理投诉时,AI应用架构师该怎么办?这5个解决步骤
  • 探秘!宜宾这5家家电门店,质量好到超乎想象!
  • Web自动化测试之数据驱动实战
  • 基于单片机的音乐盒设计
  • LobeChat抖音短视频标题优化
  • 接口测试之文件上传
  • 大模型推理基石:如何用 C++ 封装 CUDA API?(含源码与原理解析)
  • 基于大数据旅游分析可视化平台 数据大屏 游客分析+商家分析+舆情分析 Flask框架 (附源码)
  • GraphRAG:从向量检索到知识图谱,大模型推理能力的革命性突破
  • 互联网大厂Java求职者面试实录:严肃面试官与搞笑谢飞机的三轮问答
  • 大数据项目:Spark电影数据分析可视化系统 大数据 Hadoop 机器学习预测算法 爬虫 电影推荐 票房预测 猫眼电影 计算机毕业设计(全套源码+教程+开发笔记+文档)✅
  • 提示词工程:6大原则+实战案例,小白也能玩转大模型
  • 麒麟KY10系统 RPM 安装 automake-1.16.2-1.ky10.noarch 完整指南
  • 利用一句话木马夺取目标网站的shell
  • IntelliJ IDEA 使用指南
  • 构建高效RAG系统:21种文本分块策略全解析,程序员必备收藏指南
  • AI Agent全解析:从第一性原理到多Agent协作,程序员必学的大模型进阶指南
  • Jmeter 命令行压测生成HTML测试报告
  • 编程马拉松指定工具:LobeChat助力Hackathon选手
  • git rebase