news 2026/2/5 18:21:37

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

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【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)}

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/2/6 0:48:58

计算机视觉_CNN与目标检测实战

目录引言计算机视觉基础图像的数字化表示图像预处理卷积神经网络&#xff08;CNN&#xff09;基础卷积操作池化层激活函数构建完整的CNN模型目标检测基础边界框表示非极大值抑制&#xff08;NMS&#xff09;实战项目&#xff1a;简单的目标检测器数据准备简化的YOLO风格检测器训…

作者头像 李华
网站建设 2026/2/2 23:18:03

基于SpringBoot的同城宠物服务管理系统

同城宠物服务管理系统的课题背景 随着城市化进程加快和居民生活水平提高&#xff0c;宠物经济成为新兴消费热点。宠物已从单纯的看家护院角色转变为家庭重要成员&#xff0c;宠物饲养率逐年攀升&#xff0c;带动宠物食品、医疗、美容、寄养等服务需求激增。然而&#xff0c;传统…

作者头像 李华
网站建设 2026/2/6 7:10:17

【深度学习新浪潮】用AI工具解析美联储新闻,搭建量化投资分析流水线

更多分析内容,请参考我们的浮游会播客:美联储降息竟然影响你的钱包?如何把握机会、守住财富? 引言:美联储新闻+AI,解锁投资决策新范式 美联储作为全球货币政策的“锚点”,其利率决议、会议声明、官员讲话等每一条新闻都可能引发全球资产剧烈波动。但传统分析模式面临两…

作者头像 李华
网站建设 2026/2/5 10:22:01

Android 14.0 监听某个app启动或者退出功能实现

1.前言 在进行14.0的系统定制开发中,在某些app的定制过程中,需要知道某个app的启动记录和退出记录, 所以就需要监听某个app的启动和退出的过程,需要在Activity的生命周期中来实现监听功能 2.监听某个app启动或者退出功能实现的核心类 frameworks\base\core\java\android…

作者头像 李华
网站建设 2026/2/5 21:15:25

DNP3设备数据 转 IEC104项目案例

目录 1 案例说明 2 VFBOX网关工作原理 3 准备工作 4 配置网关采集DNP3协议数据 5 启用IEC104协议转发数据 6 测试网关的104功能 7 网关通过4G连接104平台 8 案例总结 1 案例说明 设置网关采集DNP3协议设备数据把采集的数据转成IEC104协议转发给其他系统。 2 VFBOX网关…

作者头像 李华
网站建设 2026/2/4 16:12:36

DDR应用专题总结

一、DDR设计之硬件设计 1.DDR硬件设计是T型结构还是非T型结构&#xff0c;直接关系到DDR能够跑的最高速率 2.DDR核电1.5v/1.8v/2.0v选择很重要&#xff0c;关系到DDR速率是否能够跑高二、MIG复位 1.mig核的init_cmpl概率性性起不来&#xff0c;需要在逻辑中设计一个复位&#x…

作者头像 李华