浅谈GO中的Channel以及死锁的造成
本篇文章向大家介绍《浅谈GO中的Channel以及死锁的造成》,主要包括GOChannel、死锁,具有一定的参考价值,需要的朋友可以参考一下。
写在前面
这篇文章的诞生要感谢MIT 6.284课程。在其中一节课中,谈到了多线程的协同的一些问题,其中就涉及到了channel这个概念,并由一段代码引发思考并逐渐深入得到了这篇文章。
引子
课程中有一段代码如下:
其大致含义是:代码背景是在进行多线程网络爬虫页面url,master线程启动后,从channel通道中读取当前页面的所有url即urls,接着再对这个urls中的每一个url进行爬虫读取新页面中的urls(即执行go worker(u, ch ,fetcher)),每启动一个worker线程便开始向channel中写入该url指向页面中所有包含的urls,以供master线程读取。
问题抛出
那么问题来了,为什么第一层for循环不会range完ch之后便直接结束循环,还需要利用局部变量n来根据特定情况跳出循环?
问题解释
课程上的解释是,这个range会一直阻塞,但并未提出解释。其实,这里很容易分析,因为当前的channel是一个无缓冲通道。所谓无缓冲通道,简单的讲就是两个线程对channel进行操作,一个读,一个写,永远都只能是写一个,读一个按照这样的顺序进行。更详细一些的话,读的那个线程会一直阻塞,直到写的线程向channel中写入一个数据。反之亦然,写的线程在完成一次写操作之后,也会一直阻塞直到另外一个线程完成对该channel的读取操作。上述情况只有一种例外状况,那就是该channel通道被某个线程close掉了:close(channel)。
而这里的range其实不太等同于对数组的range,这里的range实质上为对channel通道的读取。所以,在并未有认为close通道的前提下,该for循环会一直阻塞,不会退出,于是需要设定一个局部状态量n让其退出循环,保证程序的正常运行。当然我们也可以通过close其channel来实现,不过我认为close的时机可能不是非常容易把握。
继续深入
完成上述思考之后,对channel进行了较为的深入的分析,当然分析是以具体的实验展开的。给出下述实验代码:
func main() { test() } func test() { ch := make(chan int,4) go func() { ch <p>执行结果直接报错,显示:fatal error: all goroutines are asleep - deadlock!<br>即:出现死锁。<br>为什么会出现这种情况?<br>首先我们来分析一下这段代码的目的:利用channel通道,实现数据的传递,一个线程向channel通道中写入数据,另外一个读取。为什么会出现死锁呢?</p> <p>首先我们分析一下当前程序有多少个线程在执行,main函数是主线程,调用test函数之后,主线程进入了test函数中继续运行。而在test函数中,采用闭包函数或者说匿名函数的方法新开了一个线程,即goroutine去向已经生成的无缓冲通道中发送数据。发送的过程并非是主线程的任务,所以主线程在执行完go func之后马上跳过继续执行下面的for循环,也就是要将channel中的数据读取出来。</p> <pre class="brush:plain;">for a := range ch { fmt.Print(a) }
这时,问题来了。现在两个线程,主线程读,另外一个写。在另外一个线程完成最后一个写之后,主线程开始阻塞等待新的写操作,而主线程一旦阻塞整个test函数也无法结束,所以导致了死锁的产生,主线程一直被阻塞。
明白了上述原因之后,解决方法便很简单了,将从channel中读数据的任务交给另外一个线程,而非主线程,主线程直接调用完test函数之后马上结束,其他两个线程的死活都不会影响到程序本身的运行,即主线程的运行。如下:
func main() { test() } func test() { ch := make(chan int,4) go func() { ch <p>当然这种方法是偷懒的,这样的操作有可能导致内存溢出等情况发生,所以最好还是让发送数据的线程在发送完之后将channel关闭,如下所示:</p> <pre class="brush:plain;">func main() { test() time.Sleep(time.Second) } func test() { ch := make(chan int,4) go func() { ch <p>输出为:</p> <blockquote><p>test is over1234</p></blockquote> <p>注意,这里为了保证能够输出1234,需要将主线程休眠1s,确保主线程在退出之前,负责读取的线程能够完成读取工作。</p> <h2>写在后面</h2> <p>Go语言对多线程天然的集成性,让其在处理并发的一些事务时十分方便,但是还是需要注意一些死锁的生成。</p> <p>本篇关于《浅谈GO中的Channel以及死锁的造成》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!</p>

- 上一篇
- Go语言实现一个简单的并发聊天室的项目实战

- 下一篇
- golangcroncli定时器命令详解
-
- 儒雅的咖啡豆
- 太细致了,收藏了,感谢up主的这篇技术贴,我会继续支持!
- 2023-02-17 14:52:29
-
- 阔达的路人
- 这篇文章真及时,博主加油!
- 2023-02-02 06:14:47
-
- 深情的酒窝
- 这篇文章内容真及时,太全面了,赞 ??,已收藏,关注大佬了!希望大佬能多写Golang相关的文章。
- 2023-01-17 05:16:39
-
- 无奈的御姐
- 很好,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢大佬分享文章!
- 2023-01-17 05:09:34
-
- Golang · Go教程 | 27分钟前 |
- Golang与Terraform云资源管理实战
- 134浏览 收藏
-
- Golang · Go教程 | 37分钟前 |
- Golangos库文件与目录操作全解析
- 164浏览 收藏
-
- Golang · Go教程 | 41分钟前 |
- Golangmap教程:键值对存储与安全访问方法
- 463浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- Golang排序技巧与数据对比方法
- 103浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- Golang RPC错误码定义与处理方法
- 298浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- Golang监控集成Prometheus教程
- 319浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- GolangJSON序列化反序列化全解析
- 355浏览 收藏
-
- Golang · Go教程 | 1小时前 |
- GoTour示例运行失败解决方法
- 285浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 514次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 499次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- AI Mermaid流程图
- SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
- 622次使用
-
- 搜获客【笔记生成器】
- 搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
- 628次使用
-
- iTerms
- iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
- 644次使用
-
- TokenPony
- TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
- 711次使用
-
- 迅捷AIPPT
- 迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
- 608次使用
-
- 详解Golang并发操作中常见的死锁情形
- 2023-02-16 167浏览
-
- Go 语言中的死锁问题解决
- 2023-01-07 224浏览
-
- golang coroutine 的等待与死锁用法
- 2023-01-07 296浏览
-
- update.where无索引导致MySQL死锁问题解决
- 2022-12-31 270浏览
-
- golang 在这段代码中未检测到deadlock
- 2023-01-23 360浏览