Go并发编程之死锁与活锁的案例分析
一分耕耘,一分收获!既然都打开这篇《Go并发编程之死锁与活锁的案例分析》,就坚持看下去,学下去吧!本文主要会给大家讲到死锁、活锁等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新Golang相关的内容,希望对大家都有所帮助!
什么是死锁、活锁
什么是死锁:就是在并发程序中,两个或多个线程彼此等待对方完成操作,从而导致它们都被阻塞,并无限期地等待对方完成。这种情况下,程序会卡死,无法继续执行。
什么是活锁:就是程序一直在运行,但是无法取得进展。例如,在某些情况下,多个线程会争夺同一个资源,然后每个线程都会释放资源,以便其他线程可以使用它。但是,如果没有正确的同步,这些线程可能会同时尝试获取该资源,然后再次释放它。这可能导致线程在无限循环中运行,却无法取得进展。
发生死锁的案例分析
1.编写会发生死锁的代码:
package main import ( "fmt" "sync" ) func main() { var mu sync.Mutex mu.Lock() defer mu.Unlock() wg := sync.WaitGroup{} wg.Add(1) go func() { fmt.Println("goroutine started") mu.Lock() // 在这里获取了锁 fmt.Println("goroutine finished") mu.Unlock() wg.Done() }() wg.Wait() }
运行和输出:
[root@workhost temp02]# go run main.go
goroutine started
fatal error: all goroutines are asleep - deadlock! # 错误很明显了,告诉你死锁啦!
goroutine 1 [semacquire]:
sync.runtime_Semacquire(0xc000010030?)
/usr/local/go/src/runtime/sema.go:62 +0x27
...
...
上面的代码,使用 sync.Mutex 实现了一个互斥锁。主 goroutine 获取了锁,并启动了一个新的 goroutine。新 goroutine 也尝试获取锁来执行其任务。但是,由于主 goroutine 没有释放锁,新 goroutine 将一直等待锁,导致死锁。
2.代码改造
在上面的代码中,可以通过将主 goroutine 中的 defer mu.Unlock() 移到 goroutine 函数中的 mu.Unlock() 后面来解决问题。这样,当 goroutine 获取到锁后,它可以在完成任务后释放锁,以便主 goroutine 可以继续执行。
改造后的代码:
package main import ( "fmt" "sync" ) func main() { var mu sync.Mutex mu.Lock() wg := sync.WaitGroup{} wg.Add(1) go func() { fmt.Println("goroutine started") mu.Lock() // 在这里获取了锁 fmt.Println("goroutine finished") mu.Unlock() wg.Done() }() mu.Unlock() // 释放锁 wg.Wait() }
运行和输出:
[root@workhost temp02]# go run main.go
goroutine started
goroutine finished
3.如何避免死锁
在 Go 语言中,要避免死锁,一定要清楚以下几个规则:
- 避免嵌套锁:在使用多个锁时,确保它们的嵌套顺序相同。否则,可能会出现循环等待的情况,导致死锁。
- 避免无限等待:如果在获取锁时指定了超时时间,确保在超时后能够处理错误或执行其他操作。
- 避免过度竞争:如果多个协程需要访问相同的资源,请确保它们不会互相干扰。可以使用互斥锁或读写锁等机制来解决竞争问题。
- 使用通道:Go 语言中的通道可以用于协调并发操作。使用通道来传递消息和同步操作,可以避免死锁和竞争问题。
- 确保资源释放:在使用锁或其他资源时,一定要确保它们在使用后得到释放,否则可能会导致死锁。
- 使用 select 语句:在使用通道进行并发操作时,可以使用 select 语句来避免死锁。通过 select 语句选择多个通道中的一个进行操作,可以避免在某个通道被阻塞时出现死锁。
发生活锁的案例分析
1.编写会发生活锁的代码:
package main import ( "fmt" "sync" ) func main() { var wg sync.WaitGroup var mu sync.Mutex var flag bool wg.Add(2) // goroutine 1 go func() { // 先获取锁资源 fmt.Println("goroutine 1 获取 mu") mu.Lock() defer mu.Unlock() // 然后等待 flag 变量的值变为 true fmt.Println("goroutine 1 等待标志") for !flag { // 不断循环等待 } // 最终输出并释放锁资源 fmt.Println("goroutine 1 从等待中释放") wg.Done() }() // goroutine 2 go func() { // 先获取锁资源 fmt.Println("goroutine 2 获取 mu") mu.Lock() defer mu.Unlock() // 然后等待 flag 变量的值变为 true fmt.Println("GoRoutine2 等待标志") for !flag { // 不断循环等待 } // 最终输出并释放锁资源 fmt.Println("GoRoutine 2 从等待中释放") wg.Done() }() // 在主线程中等待 1 秒钟,以便两个 goroutine 开始等待 flag 变量的值 // 然后将 flag 变量设置为 true // 由于两个 goroutine 会同时唤醒并尝试获取锁资源,它们会相互等待 // 最终导致了活锁问题,它们都无法向前推进 fmt.Println("主线程休眠 1 秒") fmt.Println("两个goroutine都应该等待标志") flag = true wg.Wait() fmt.Println("所有 GoRoutines 已完成") }
运行和输出:
[root@workhost temp02]# go run main.go
主线程休眠 1 秒
两个goroutine都应该等待标志
goroutine 2 获取 mu
GoRoutine2 等待标志
GoRoutine 2 从等待中释放
goroutine 1 获取 mu
goroutine 1 等待标志
goroutine 1 从等待中释放
所有 GoRoutines 已完成
上面的代码存在活锁问题。如果两个goroutine同时等待flag变为true并且都已经获取了锁资源,那么它们就会进入一个死循环并相互等待,无法继续向前推进。
2.代码改造
改造后的代码:
package main import ( "fmt" "runtime" "sync" ) func main() { var wg sync.WaitGroup var mu sync.Mutex var flag bool wg.Add(2) // goroutine 1 go func() { // 先获取锁资源 fmt.Println("goroutine 1 获取 mu") mu.Lock() defer mu.Unlock() // 然后等待 flag 变量的值变为 true fmt.Println("goroutine 1 等待标志") for !flag { runtime.Gosched() // 让出时间片 } // 最终输出并释放锁资源 fmt.Println("goroutine 1 从等待中释放") wg.Done() }() // goroutine 2 go func() { // 先获取锁资源 fmt.Println("goroutine 2 获取 mu") mu.Lock() defer mu.Unlock() // 然后等待 flag 变量的值变为 true fmt.Println("GoRoutine2 等待标志") for !flag { runtime.Gosched() // 让出时间片 } // 最终输出并释放锁资源 fmt.Println("GoRoutine 2 从等待中释放") wg.Done() }() // 在主线程中等待 1 秒钟,以便两个 goroutine 开始等待 flag 变量的值 // 然后将 flag 变量设置为 true // 由于两个 goroutine 会同时唤醒并尝试获取锁资源,它们会相互等待 // 最终导致了活锁问题,它们都无法向前推进 fmt.Println("主线程休眠 1 秒") fmt.Println("两个goroutine都应该等待标志") flag = true wg.Wait() fmt.Println("所有 GoRoutines 已完成") }
改造后的代码在等待flag变量的循环中加入了让出时间片的函数 runtime.Gosched(),这样两个goroutine在等待期间可以放弃时间片,以便其他goroutine可以执行并获得锁资源。这种方式可以有效地减少竞争程度,从而避免了活锁问题。
3.如何避免发生活锁的可能性
在 Go 语言的并发编程中,避免活锁的关键是正确地实现同步机制。以下是一些避免活锁的方法:
- 避免忙等待:使用 sync.Cond 或者 channel 等同步机制来实现等待。这样避免了线程一直占用 CPU 资源而无法取得进展的问题。
- 避免死锁:死锁往往是活锁的前提,因此正确地使用锁和同步机制可以避免死锁,从而避免活锁。
- 减少锁的粒度:尽可能将锁的粒度缩小到最小范围,避免锁住不必要的代码块。
- 采用超时机制:使用 sync.Mutex 的 TryLock() 方法或者使用 select 语句实现等待超时机制,这样可以防止线程无限期等待。
- 合理设计并发模型:合理设计并发模型可以避免竞争和饥饿等问题,进而避免活锁的发生。
终于介绍完啦!小伙伴们,这篇关于《Go并发编程之死锁与活锁的案例分析》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

- 上一篇
- 深入探究Golang中flag标准库的使用

- 下一篇
- Jackson2JsonRedisSerializer和GenericJackson2JsonRedisSerializer区别
-
- Golang · Go教程 | 1天前 |
- TigervncDebian多用户共享桌面超简单教程
- 482浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Go语言新手必看!切片vs数组,一次搞定这两个核心知识点
- 472浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Docker在Debian上运行超简单教程(保姆级教学)
- 210浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian设置hostname踩坑记录:权限问题大揭秘
- 334浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian装SQLServer?这些问题你一定要注意!
- 284浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian系统下Jenkins自动化部署脚本教学
- 367浏览 收藏
-
- Golang · Go教程 | 2天前 |
- DebianSwap服务器应用实测,这些场景真的用得上!
- 319浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian跑TigerVNC实测!真香警告,快来看看性能咋样~
- 171浏览 收藏
-
- Golang · Go教程 | 2天前 |
- 在Debian上玩转SQLServer备份还原,手把手教你一步步操作
- 498浏览 收藏
-
- Golang · Go教程 | 2天前 |
- DebianOverlay不会玩?手把手教你轻松定制化安装
- 258浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Go语言实战:time.Ticker&time.After用法区别及避坑技巧
- 240浏览 收藏
-
- Golang · Go教程 | 2天前 |
- Debian系统如何快速定位&干掉那些讨厌的僵尸进程
- 317浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 茅茅虫AIGC检测
- 茅茅虫AIGC检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
- 25次使用
-
- 赛林匹克平台(Challympics)
- 探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
- 50次使用
-
- 笔格AIPPT
- SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
- 58次使用
-
- 稿定PPT
- 告别PPT制作难题!稿定PPT提供海量模板、AI智能生成、在线协作,助您轻松制作专业演示文稿。职场办公、教育学习、企业服务全覆盖,降本增效,释放创意!
- 54次使用
-
- Suno苏诺中文版
- 探索Suno苏诺中文版,一款颠覆传统音乐创作的AI平台。无需专业技能,轻松创作个性化音乐。智能词曲生成、风格迁移、海量音效,释放您的音乐灵感!
- 60次使用
-
- 浅谈GO中的Channel以及死锁的造成
- 2022-12-30 462浏览
-
- 详解Golang并发操作中常见的死锁情形
- 2023-02-16 167浏览
-
- Go 语言中的死锁问题解决
- 2023-01-07 224浏览
-
- golang coroutine 的等待与死锁用法
- 2023-01-07 296浏览
-
- update.where无索引导致MySQL死锁问题解决
- 2022-12-31 270浏览