GolangMutex互斥锁深入理解
本篇文章给大家分享《GolangMutex互斥锁深入理解》,覆盖了Golang的常见基础知识,其实一个语言的全部知识点一篇文章是不可能说完的,但希望通过这些问题,让读者对自己的掌握程度有一定的认识(B 数),从而弥补自己的不足,更好的掌握它。
引言
Golang的并发编程令人着迷,使用轻量的协程、基于CSP的channel、简单的go func()就可以开始并发编程,在并发编程中,往往离不开锁的概念。
本文介绍了常用的同步原语 sync.Mutex,同时从源码剖析它的结构与实现原理,最后简单介绍了mutex在日常使用中可能遇到的问题,希望大家读有所获。
Mutex结构
Mutex运行时数据结构位于sync/mutex.go包
type Mutex struct {
state int32
sema uint32
}
其中state表示当前互斥锁的状态,sema表示 控制锁状态的信号量.
互斥锁的状态定义在常量中:
const ( mutexLocked = 10即其他状态。
sema是一个组合,低三位分别表示锁的三种状态,高29位表示正在等待互斥锁释放的gorountine个数,和Java表示线程池状态那部分有点类似
一个mutex对象仅占用8个字节,让人不禁感叹其设计的巧妙
饥饿模式和正常模式
正常模式
在正常模式下,等待的协程会按照先进先出的顺序得到锁 在正常模式下,刚被唤醒的goroutine与新创建的goroutine竞争时,大概率无法获得锁。
饥饿模式
为了避免正常模式下,goroutine被“饿死”的情况,go在1.19版本引入了饥饿模式,保证了Mutex的公平性
在饥饿模式中,互斥锁会直接交给等待队列最前面的goroutine。新的goroutine 在该状态下不能获取锁、也不会进入自旋状态,它们只会在队列的末尾等待。
状态的切换
在正常模式下,一旦Goroutine超过1ms没有获取到锁,它就会将当前互斥锁切换饥饿模式
如果一个goroutine 获得了互斥锁并且它在队列的末尾或者它等待的时间少于 1ms,那么当前的互斥锁就会切换回正常模式。
加锁和解锁
加锁
func (m *Mutex) Lock() { // Fast path: grab unlocked mutex. if atomic.CompareAndSwapInt32(&m.state, 0, mutexLocked) { return } // 原注释: Slow path (outlined so that the fast path can be inlined) // 将 m.lockSlow() }可以看到,当前互斥锁的状态为0时,尝试将当前锁状态设置为更新锁定状态,且这些操作是原子的。
若当前状态不为0,则进入
lockSlow方法
先定义了几个参数var waitStartTime int64 starving := false // awoke := false iter := 0 old := m.state随后进入一个很大的for循环,让我们来逐步分析
自旋
for { // 1 && 2 if old&(mutexLocked|mutexStarving) == mutexLocked && runtime_canSpin(iter) { // 3. if !awoke && old&mutexWoken == 0 && old>>mutexWaiterShift != 0 && atomic.CompareAndSwapInt32(&m.state, old, old|mutexWoken) { awoke = true } runtime_doSpin() iter++ old = m.state continue }old&(mutexLocked|mutexStarving) == mutexLocked
当且仅当当前锁状态为mutexLocked时,表达式为true
runtime_canSpin(iter)是否满足自旋条件
- 运行在拥有多个CPU的机器上;
- 当前Goroutine为了获取该锁进入自旋的次数小于四次;
- 当前机器上至少存在一个正在运行的处理器 P,并且处理的运行队列为空;
如果当前状态下自旋是合理的,将awoke置为true,同时设置锁状态为mutexWoken,进入自旋逻辑
runtime_doSpin()会执行30次PAUSE指令,并且仅占用CPU资源 代码位于:runtime\asm_amd64.s +567
//go:linkname sync_runtime_doSpin sync.runtime_doSpin
//go:nosplit
func sync_runtime_doSpin() {
procyield(active_spin_cnt)
}
TEXT runtime·procyield(SB),NOSPLIT,$0-0
MOVL cycles+0(FP), AX
again:
PAUSE
SUBL $1, AX
JNZ again
RET
计算锁的新状态
停止了自旋后,
new := old
// 1.
if old&mutexStarving == 0 {
new |= mutexLocked
}
// 2.
if old&(mutexLocked|mutexStarving) != 0 {
new += 1
old&mutexStarving == 0表明原来不是饥饿模式。如果是饥饿模式的话,其他goroutine不会执行接下来的代码,直接进入等待队列队尾- 如果原来是
mutexLocked或者mutexStarving模式,waiterCounts数加一 - 如果被标记为饥饿状态,且锁状态为
mutexLocked的话,设置锁的新状态为饥饿状态。 - 被标记为饥饿状态的前提是
被唤醒过且抢锁失败 - 计算新状态
更新锁状态
// 1.
if atomic.CompareAndSwapInt32(&m.state, old, new) {
if old&(mutexLocked|mutexStarving) == 0 {
break // locked the mutex with CAS
}
// 2.
queueLifo := waitStartTime != 0
if waitStartTime == 0 {
waitStartTime = runtime_nanotime()
}
// 3.
runtime_SemacquireMutex(&m.sema, queueLifo, 1)
// 4.
starving = starving || runtime_nanotime()-waitStartTime > starvationThresholdNs
old = m.state
// 5.
if old&mutexStarving != 0 {
/
if old&(mutexLocked|mutexWoken) != 0 || old>>mutexWaiterShift == 0 {
throw("sync: inconsistent mutex state")
}
delta := int32(mutexLocked - 1>mutexWaiterShift == 1 {
delta -= mutexStarving
}
atomic.AddInt32(&m.state, delta)
break
}
awoke = true
iter = 0
} else {
old = m.state
}
}
- 尝试将锁状态设置为new 。这里设置成功不代表上锁成功,有可能new不为
mutexLocked或者是waiterCount数量的改变 waitStartTime不为0 说明当前goroutine已经等待过了,将当前goroutine放到等待队列的队头- 走到这里,会调用
runtime_SemacquireMutex方法使当前协程阻塞,runtime_SemacquireMutex方法中会不断尝试获得锁,并会陷入休眠 等待信号量释放。 - 当前协程可以获得信号量,从
runtime_SemacquireMutex方法中返回。此时协程会去更新starving标志位:如果当前starving标志位为true或者等待时间超过starvationThresholdNs,将starving置为true
之后会按照饥饿模式与正常模式,走不同的逻辑
- - 在正常模式下,这段代码会设置唤醒和饥饿标记、重置迭代次数并重新执行获取锁的循环;
- - 在饥饿模式下,当前 Goroutine 会获得互斥锁,如果等待队列中只存在当前 Goroutine,互斥锁还会从饥饿模式中退出;
解锁

func (m *Mutex) Unlock() {
// 1.
new := atomic.AddInt32(&m.state, -mutexLocked)
if new != 0 {
// 2.
m.unlockSlow(new)
}
}
- 将锁状态的值增加 -mutexLocked 。如果新状态不等于0,进入
unlockSlow方法
func (m *Mutex) unlockSlow(new int32) {
// 1.
if (new+mutexLocked)&mutexLocked == 0 {
throw("sync: unlock of unlocked mutex")
}
if new&mutexStarving == 0 {
old := new
for {
// 2.
if old>>mutexWaiterShift == 0 || old&(mutexLocked|mutexWoken|mutexStarving) != 0 {
return
}
// 2.1.
new = (old - 1
1.new+mutexLocked代表将锁置为1,如果两个状态& 不为0,则说明重复解锁.如果重复解锁则抛出panic
2. 如果等待者数量等于0,或者锁的状态已经变为mutexWoken、mutexStarving、mutexStarving,则直接返回
- 将waiterCount数量-1,尝试选择一个goroutine唤醒
- 尝试更新锁状态,如果更新锁状态成功,则唤醒队尾的一个gorountine
3. 如果不满足 2的判断条件,则进入饥饿模式,同时交出锁的使用权
可能遇到的问题
锁拷贝
mu1 := &sync.Mutex{}
mu1.Lock()
mu2 := mu1
mu2.Unlock()
此时mu2能够正常解锁,那么我们再试试解锁mu1呢
mu1 := &sync.Mutex{}
mu1.Lock()
mu2 := mu1
mu2.Unlock()
mu1.Unlock()

可以看到发生了error
panic导致没有unlock
当lock()之后,可能由于代码问题导致程序发生了panic,那么mutex无法被及时unlock(),由于其他协程还在等待锁,此时可能触发死锁
func TestWithLock() {
nums := 100
wg := &sync.WaitGroup{}
safeSlice := SafeSlice{
s: []int{},
lock: new(sync.RWMutex),
}
i := 0
for idx := 0; idx

修改:
func TestWithLock() {
nums := 100
wg := &sync.WaitGroup{}
safeSlice := SafeSlice{
s: []int{},
lock: new(sync.RWMutex),
}
i := 0
for idx := 0; idx
今天关于《GolangMutex互斥锁深入理解》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于golang的内容请关注golang学习网公众号!
Golang信号量设计实现示例详解
- 上一篇
- Golang信号量设计实现示例详解
- 下一篇
- golang 防缓存击穿singleflight的实现
-
- 合适的春天
- 太详细了,收藏了,感谢老哥的这篇文章,我会继续支持!
- 2023-04-03 03:48:09
-
- 阳光的自行车
- 这篇技术贴出现的刚刚好,作者加油!
- 2023-03-30 17:22:22
-
- 迷路的小鸽子
- 这篇技术文章真是及时雨啊,太详细了,赞 ??,码住,关注up主了!希望up主能多写Golang相关的文章。
- 2023-02-01 16:12:52
-
- 风趣的天空
- 受益颇多,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢作者分享博文!
- 2023-01-25 20:49:46
-
- 腼腆的楼房
- 这篇技术贴真及时,细节满满,太给力了,码起来,关注作者了!希望作者能多写Golang相关的文章。
- 2023-01-06 01:48:35
-
- 激昂的小虾米
- 感谢大佬分享,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢楼主分享文章内容!
- 2023-01-04 17:02:58
-
- 还单身的康乃馨
- 这篇文章内容真是及时雨啊,太详细了,很好,码住,关注up主了!希望up主能多写Golang相关的文章。
- 2023-01-03 09:29:22
-
- 开放的白昼
- 这篇文章内容真及时,太详细了,很有用,mark,关注大佬了!希望大佬能多写Golang相关的文章。
- 2022-12-30 19:14:34
-
- Golang · Go教程 | 17小时前 | goroutine · Context · 超时控制 · Go教程 · 后端开发 · Go Goroutine context 超时控制 WithTimeout Done QueryContext
- Go context 超时控制实战:从接口入口到 goroutine 回收的完整流程
- 166浏览 收藏
-
- Golang · Go教程 | 2天前 | map · 并发安全 · RWMutex · sync.Map · Go教程 · 并发安全 RWMutex sync.Map Go map并发读写 go test race
- Go map 并发读写崩溃怎么办:从复现报错到 RWMutex 修复的完整流程
- 272浏览 收藏
-
- Golang · Go教程 | 4天前 | singleflight · 并发控制 · Go教程 · 缓存治理 · 接口优化 · Go 并发请求 缓存击穿 singleflight 缓存回填
- Go singleflight 防缓存击穿实战:相同请求只查一次数据库
- 114浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ljg-skills
- ljg-skills 是李继刚开源的 AI 技能与提示词集合,面向大模型使用者整理了一批可复用的 prompt、角色设定和任务技能模板,适合用于学习提示词设计、搭建个人 AI 工作流和沉淀团队常用智能体能力。
- 542次使用
-
- MELO音乐
- MELO音乐是一站式AI视频与音乐制作助手,对标suno, udio的高品质体验。提供伴奏生成、原创写词、无损导出、哼唱识曲、混音变声等全套音频与短视频编辑工具。无论是流行Kpop、电音说唱、民谣古风、摇滚儿歌还是商用轻音乐,MELO为你免费谱曲,轻松做同款!
- 554次使用
-
- UniScribe
- UniScribe 是一款 AI 音视频转文字与内容整理工具,支持上传音频、视频文件或粘贴 YouTube 链接,自动生成转写文本、摘要、思维导图和关键问题,并支持多格式导出,适合会议记录、课程学习、访谈整理和内容创作复盘。
- 511次使用
-
- 剧云
- 剧云是专业中文剧本创作平台,安全稳定运行十余年,集成AI编剧、剧本医生审核、人物小传、剧情关系图、大纲编写、多人协作、Word导入导出、版权管控功能,数据安全防护,轻松高效创作剧本。
- 687次使用
-
- 万象有声
- 万象有声,一个专为有声创作者打造的新一代智能有声内容创作平台。平台提供专业的智能拆章、智能画本编辑、AI配音、AI生成音效、后期制作、智能对轨、智能审听等有声创作全流程工具,可以帮助创作者高效、低成本创作出引人入胜的有声作品。立即体验,让有声书制作更简单!
- 667次使用
-
- Golang Mutex互斥锁源码分析
- 2022-12-22 200浏览
-
- 初识Golang Mutex互斥锁的使用
- 2022-12-22 312浏览
-
- Go与Redis实现分布式互斥锁和红锁
- 2022-12-22 117浏览
-
- Golang Mutex 原理详细解析
- 2022-12-28 439浏览
-
- Go语言底层原理互斥锁的实现原理
- 2022-12-28 446浏览



