当前位置: 首页 > 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.zskr.cn/news/112677.html

相关文章:

  • LobeChat公益活动策划方案生成
  • 探秘!宜宾这5家家电门店,质量好到超乎想象!
  • 大模型推理基石:如何用 C++ 封装 CUDA API?(含源码与原理解析)
  • 基于大数据旅游分析可视化平台 数据大屏 游客分析+商家分析+舆情分析 Flask框架 (附源码)
  • GraphRAG:从向量检索到知识图谱,大模型推理能力的革命性突破
  • 构建高效RAG系统:21种文本分块策略全解析,程序员必备收藏指南
  • AI Agent全解析:从第一性原理到多Agent协作,程序员必学的大模型进阶指南
  • Jmeter 命令行压测生成HTML测试报告
  • 编程马拉松指定工具:LobeChat助力Hackathon选手
  • 软著提交时人数过多系统繁忙问题,终极解决办法!
  • AI编程系列——mcp与skill
  • 基于单片机的交通红绿灯控制系统
  • Netcode for GameObjects Boss Room 多人RPG战斗(7)
  • TensorFlow损失函数的“隐形坑”
  • LobeChat技术面试题生成器开发
  • AI之 n8n
  • 大数据Python招聘推荐系统 数据分析可视化 Django+可视化+协同过滤算法 毕业设计(附源码+文档)
  • 生日祝福个性化:LobeChat记住每个人的喜好
  • Java毕设项目:基于Javaweb的二手儿童绘本交易系统设计与实现(源码+文档,讲解、调试运行,定制等)
  • Java毕设项目:基于java的城市公交调度系统(源码+文档,讲解、调试运行,定制等)
  • Netcode for GameObjects Boss Room 多人RPG战斗(1)
  • 基于springboot物流管理系统毕业论文+PPT(附源代码+演示视频)
  • 室友分享的7个降AI工具,论文ai率从80%降低到13%!
  • AIGC率90%怎么降低ai?测7款降a率工具,降AI效果嘎嘎好!
  • AIGC率80%怎么降低ai?7个主流降AI率工具,降AI效果嘎嘎好!
  • LobeChat能否取代商业聊天界面?对比测试结果出炉
  • 不敢相信!这5个良心软件,功能强大到媲美付费版!
  • 大语言模型(LLM)技能需求激增300%,这条学习路径让你从新手变身AI应用构建高手!
  • 腾讯云国际站代理商的TAPD如何帮助企业进行研发管理?
  • 节日贺卡设计:LobeChat生成温馨祝福语