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

002 Golang-channel-practice

第二题:

创建一个生产器和接收器,再建立一个无缓冲的channel。生产器负责把数据放进管道里,接收器负责把管道里面的数据打印出来。这里我们开5个协程把数据打印出来。

直接上代码!

package mainimport ("fmt"
)func receive(c <-chan int) {/*for v := range c {fmt.Println("received:", v)}*/for i := 0; i <= 1; i++ {go func() {for v := range c {fmt.Println(v)}}()}
}
func generator() <-chan int {c := make(chan int)for i := 0; i <= 9; i++ {go func(i int) {for j := 0; j <= 9; j++ {temp := i*100 + 20 + jc <- temp}close(c)}(i)}return c
}
func main() {c := generator()receive(c)
}

埋了个小坑,跑上面的代码,在这里是不会有任何输出的。

87349339b46d4558b9a0c71f976ee5a4.png

原因是main函数结束时程序就退出了,没有给goroutine足够运行的时间来打印输出。

整个流程是并发执行的,main函数、generator的goroutine、receive的goroutine都是并发运行。

但是问题是main函数和generator很快就结束了,程序退出,receive的goroutine来不及打印数据。

解决方法就是让main函数等一等receive的goroutine。我们在main函数中加上一句:

time.Sleep(time.Second * 5) 

这时看到可以顺利输出了。

但是...

f0e9170ff1914f46baf6820c0b15c56c.png

但是却panic了。为什么呢?

因为generator()把消息发送到了关闭的管道。是因为生成器goroutine和接收goroutine的生命周期没有控制好导致的。

主要原因在于,接收的goroutine一旦从通道接收完所有的数据并退出,通道就会被关闭。

而此时,生成器goroutine可能还在向这个通道发送数据,于是产生了panic。

要避免这种情况,需要确保:

 

1、接收goroutine在最后一个生成器goroutine退出之前不能退出。

2、生成器goroutine在关闭通道之前,必须保证接收goroutine仍在运行。

 

问题出在生成器中close(c)这一行。这里每个goroutine都在自己完成后关闭了通道c。

按照程序逻辑,通道c应该在最后一个goroutine完成时关闭一次,而不是每个goroutine都关闭。所以应该只在主goroutine中关闭c。这里我们用WaitGroup来同步。

 

func generator() <-chan int {c := make(chan int) var wg sync.WaitGroupwg.Add(10) // 添加10个goroutinefor i := 0; i < 10; i++ {go func() {// 生成数据 wg.Done() // goroutine结束}()} go func() {wg.Wait()   // 等待所有goroutine完成close(c) // 关闭通道,仅关闭一次 }()return c 
}

 

顺利输出!!

e5ee8c796d6e454c9e30b504158688a3.png

 

 

http://www.lryc.cn/news/281369.html

相关文章:

  • MFC为对话框资源添加类
  • SpringBoot新手入门完整教程和项目示例
  • PHP留言板实现
  • ssm+vue的物流配送人员车辆调度管理系统的设计与实现(有报告)。Javaee项目,ssm vue前后端分离项项目。
  • day1·算法-双指针
  • 在vue中,切换页面之后如何关闭定时器
  • 观测云产品更新 | 日志、场景仪表板、监控器等
  • 【JupyterLab】在 conda 虚拟环境中 JupyterLab 的安装与使用
  • HTML--JavaScript--引入方式
  • 第28关 k8s监控实战之Prometheus(七)
  • SSC | Blue Prism报告:2024年智能自动化(IA)7大趋势预测
  • el-tree定义左边箭头,包括下级出现连线
  • C++ 多线程顺序打印
  • x-cmd pkg | duf - df 命令的现代化替代品
  • 202406读书笔记|《沉睡的线条世界》——翻山越岭,只为与你分享点滴的快乐
  • [论文阅读]4DRadarSLAM: A 4D Imaging Radar SLAM System for Large-scale Environments
  • Python: vars()详细解释
  • 2024年1月15日Arxiv最热论文推荐:斯坦福LLM精准微调新框架、GPT不愿承认回答错误、速度快15倍的3D全景分割新突破
  • 1.5 面试经典150题 - 轮转数组
  • Linux的基础命令学习
  • 个人数据备份方案分享(源自一次悲惨经历)
  • SpringBoot教程(八) | SpringBoot统一结果封装
  • Ubuntu 22.04 安装Fail2Ban
  • Ubuntu 22.04 编译安装 Qt mysql驱动
  • Mindspore 公开课 - CodeGeeX
  • 说一下mysql的锁
  • rime中州韵小狼毫 日期/农历 时间 事件 节气 滤镜
  • 【前端】前后端的网络通信基础操作(原生ajax, axios, fetch)
  • Matter - 配置工厂数据(2)
  • 版本控制背景知识