我的编程空间,编程开发者的网络收藏夹
学习永远不晚

详解go语言中并发安全和锁问题

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

详解go语言中并发安全和锁问题

首先可以先看看这篇文章,对锁有些了解

GO语言并发编程之互斥锁、读写锁详解

Mutex-互斥锁

Mutex 的实现主要借助了 CAS 指令 + 自旋 + 信号量

数据结构:


type Mutex struct {
	state int32
	sema  uint32
}

上述两个加起来只占 8 字节空间的结构体表示了 Go语言中的互斥锁

状态:

在默认情况下,互斥锁的所有状态位都是 0,int32 中的不同位分别表示了不同的状态:

  • 1位表示是否被锁定
  • 1位表示是否有协程已经被唤醒
  • 1位表示是否处于饥饿状态
  • 剩下29位表示阻塞的协程数

正常模式和饥饿模式

正常模式:所有goroutine按照FIFO的顺序进行锁获取,被唤醒的goroutine和新请求锁的goroutine同时进行锁获取,通常新请求锁的goroutine更容易获取锁(持续占有cpu),被唤醒的goroutine则不容易获取到锁

饥饿模式:所有尝试获取锁的goroutine进行等待排队,新请求锁的goroutine不会进行锁获取(禁用自旋),而是加入队列尾部等待获取锁

如果一个 Goroutine 获得了互斥锁并且它在队列的末尾或者它等待的时间少于 1ms,那么当前的互斥锁就会切换回正常模式。

与饥饿模式相比,正常模式下的互斥锁能够提供更好地性能,饥饿模式的能避免 Goroutine 由于陷入等待无法获取锁而造成的高尾延时。

互斥锁加锁过程

  • 如果互斥锁处于初始状态,会直接加锁
  • 如果互斥锁处于加锁状态,并且工作在普通模式下,goroutine会进入自旋,等待锁的释放

goroutine 进入自旋的条件非常苛刻:

  • 互斥锁只有在普通模式才能进入自旋;
  • runtime.sync_runtime_canSpin需要返回 true

运行在多 CPU 的机器上;

当前 Goroutine 为了获取该锁进入自旋的次数小于四次;

当前机器上至少存在一个正在运行的处理器 P 并且处理的运行队列为空;

  • 如果当前 Goroutine 等待锁的时间超过了 1ms,互斥锁就会切换到饥饿模式;
  • 互斥锁在正常情况下会通runtime.sync_runtime_SemacquireMutex将尝试获取锁的 Goroutine 切换至休眠状态,等待锁的持有者唤醒;
  • 如果当前 Goroutine 是互斥锁上的最后一个等待的协程或者等待的时间小于 1ms,那么它会将互斥锁切换回正常模式;

互斥锁解锁过程

当互斥锁已经被解锁时,再解锁会抛出异常

当互斥锁处于饥饿模式时,将锁的所有权交给等待队列最前面的 Goroutine

当互斥锁处于正常模式时,如果没有 Goroutine 等待锁的释放或者已经有被唤醒的 Goroutine 获得了锁,会直接返回;在其他情况下会通过唤醒对应的 Goroutine;

关于互斥锁锁的使用建议写业务时不能全局使用同一个 Mutex千万不要将要加锁和解锁分到两个以上 Goroutine 中进行Mutex 千万不能被复制(包括不能通过函数参数传递),否则会复制传参前锁的状态:已锁定 or 未锁定。很容易产生死锁,关键是编译器还发现不了这个 Deadlock~

RWMutex-读写锁

Go 中 RWMutex 使用的是写优先的设计

数据结构:


type RWMutex struct {
	w           Mutex	//复用互斥锁提供的能力
	writerSem   uint32	//writer信号量
	readerSem   uint32	//reader信号量
	readerCount int32	//存储了当前正在执行的读操作数量
	readerWait  int32	// 表示写操作阻塞时,等待读操作完成的个数
}

写锁

获取写锁 :

  • 调用结构体持有的Mutex结构体的Mutex.Lock阻塞后续的写操作
  • readerCount减少2^30,成为负数,以阻塞后续读操作
  • 如果有其他Goroutine 持有读锁,该 Goroutine会进入休眠状态等待所有读锁执行结束后释放writerSem信号量将当前协程唤醒

释放写锁:

  • readerCount变回正数,释放读锁
  • 唤醒所有因为读锁而睡眠的Goroutine
  • 调用Mutex.Unlock 释放写锁

获取写锁时会先阻塞写锁的获取,后阻塞读锁的获取,这种策略能够保证读操作不会被连续的写操作『饿死』。

读锁

获取读锁

获取读锁的方法 sync.RWMutex.RLock 很简单,该方法会将readerCount加一:

  • 如果该方法返回负数(代表其他 goroutine 获得了写锁,当前 goroutine 就会使其陷入休眠等待锁的释放
  • 如果该方法返回结果为非负数,代表没有 goroutine 获得写锁,会成功返回

释放读锁

解锁读锁的方法sync.RWMutex.RUnlock,该方法会:

  • readerCount减一,根据返回值的不同会分别进行处理
  • 如果返回值大于等于0,读锁直接解锁成功
  • 如果小于0代表有正在执行的写操作,会调用sync.RWMutex.rUnlockSlow,将readerWait减一,并且当所有读操作都被释放后触发信号量 writerSem,该信号量被触发时,调度器就会唤醒尝试获取写锁的 Goroutine

WaitGroup

sync.WaitGroup可以等待一组 Goroutine 的返回

sync.WaitGroup 对外暴露了三个方法:

方法名 功能
(wg * WaitGroup) Add(delta int) 计数器+delta
(wg *WaitGroup) Done() 计数器减1
(wg *WaitGroup) Wait() 阻塞直到计数器变为0

sync.WaitGroup.Done只是对 sync.WaitGroup.Add 方法的简单封装,相当于是加 -1

Sync.Map

Go语言中内置的map不是并发安全的。

Go语言的sync包中提供了一个开箱即用的并发安全版map–sync.Map。使用互斥锁保证并发安全

数据结构:


type Map struct {
    mu Mutex
    read atomic.Value // readOnly
    dirty map[interface{}]*entry
    misses int
}

开箱即用表示不用像内置的map一样使用make函数初始化就能直接使用。同时sync.Map内置了方法:

方法名 功能
(m *sync.Map)Store(key, value interface{}) 保存键值对
(m *sync.Map)Load(key interface{}) 根据key获取对应的值
(m *sync.Map)Delete(key interface{}) 删除键值对
(m *sync.Map)Range(f func(key, value interface{}) bool) 遍历 sync.Map。Range 的参数是一个函数
*sync.map 没有Len( ) 方法

原子操作(atomic包)

代码中的加锁操作因为涉及内核态的上下文切换会比较耗时、代价比较高。针对基本数据类型我们还可以使用原子操作来保证并发安全,因为原子操作是Go语言提供的方法它在用户态就可以完成,因此性能比加锁操作更好。Go语言中原子操作由内置的标准库sync/atomic提供。

参考资料:

Go 语言并发编程、同步原语与锁 | Go 语言设计与实现 (draveness.me)

到此这篇关于go语言中并发安全和锁的文章就介绍到这了,更多相关go语言中并发安全和锁内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

详解go语言中并发安全和锁问题

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

解决Go语言Websocket应用程序中的并发安全问题

WebSocket是一种现代网络通信协议,能够实现实时性很强的双向通信。Go语言天生就支持并发,因此它在Websocket应用中的表现十分出色。然而,并发也会带来一些问题,在Websocket应用程序中,这主要表现在并发安全性方面。在本文中
解决Go语言Websocket应用程序中的并发安全问题
2023-12-14

在Go语言中如何处理并发安全性问题?

在Go语言中如何处理并发安全性问题?Go语言是一门专门用于处理并发编程的语言,因此在处理并发安全性问题上具有很强的优势。在多个goroutine同时访问和修改共享数据的情况下,如果不采取适当的措施来保证并发安全性,就会导致意想不到的错误和数
2023-10-22

Go语言中如何解决并发定时器问题?

Go语言中的并发定时器问题是指在多个goroutine同时需要使用定时器时可能会出现的一些并发相关的问题。为了解决这些问题,Go语言提供了一些机制和技巧,本文将详细介绍这些解决方案,并给出代码示例。使用time.TickerGo语言的标准库
2023-10-22

如何解决Go语言中的并发文件的读写锁冲突问题?

如何解决Go语言中的并发文件的读写锁冲突问题?在Go语言中,我们经常会遇到需要同时对一个文件进行读写操作的场景,比如并发地写日志文件。如果不加以控制,多个goroutine同时对同一个文件进行读写操作,就会产生冲突,导致数据丢失或不一致。为
2023-10-22

如何解决Go语言中的并发调试问题?

如何解决Go语言中的并发调试问题?引言:并发是Go语言的一大特点,但是并发也带来了一些调试上的挑战。在多个goroutine同时执行的情况下,出现问题时很难追踪和调试错误。本文将介绍一些解决Go语言中并发调试问题的方法,并给出具体的代码示例
2023-10-22

在Go语言中如何解决并发缓存访问问题?

在Go语言中如何解决并发缓存访问问题?在并发编程中,缓存是一种常用的优化策略。通过缓存数据,可以减少对底层存储的频繁访问,提高系统的性能。然而,在多个并发访问的场景下,经常会遇到并发缓存访问问题,如缓存竞争、缓存穿透等。本文将介绍在Go语言
2023-10-22

Go语言中如何解决并发日志记录问题?

Go语言作为一种强大的编程语言,以其高效的并发性能而闻名。然而,在并发编程中,一个常见的问题是如何解决并发日志记录的问题。在这篇文章中,我们将介绍如何使用Go语言解决并发日志记录问题,并提供一些具体的代码示例。为了更好地理解并发日志记录的问
2023-10-22

Go语言中如何解决并发请求限流问题?

Go语言中如何解决并发请求限流问题?在高并发的场景下,很容易出现请求过多的情况,这会给系统带来很大的压力,甚至导致系统崩溃。因此,限制并发请求数量是必不可少的。本文将介绍如何在Go语言中解决并发请求限流问题,并提供具体的代码示例。一、什么是
2023-10-22

如何解决Go语言中的死锁问题?

如何解决Go语言中的死锁问题?Go语言具有并发编程的特性,可以通过使用goroutine和channel来实现并发操作。然而,在并发编程中,死锁是一个常见的问题。当goroutine之间相互依赖于彼此的资源,并且在访问这些资源时产生了循环依
2023-10-22

在Go语言中如何解决并发任务并行执行问题?

在Go语言中如何解决并发任务并行执行问题?在编程领域中,并发任务的并行执行是一个常见的需求。而Go语言作为一种面向并发的编程语言,为我们提供了一些强大的并发编程特性,使得解决并发任务并行执行问题变得非常简单和高效。Go语言通过gorouti
2023-10-22

Go语言中如何处理并发请求合并问题?

Go语言中如何处理并发请求合并问题?随着互联网的快速发展,处理大量并发请求成为了在开发中常常需要面对的问题。而当我们面对大量的并发请求时,为了提高系统的性能和资源利用率,我们往往需要将多个相似的请求合并处理。在Go语言中,有几种方法可以处理
2023-10-22

在Go语言中如何解决并发任务并行执行问题

在Go语言中,可以使用goroutine和channel来解决并发任务并行执行的问题。首先,使用关键字go来创建一个goroutine,将需要并发执行的任务放在其中。例如:```gogo func() {// 并发执行的任务}()```然后
2023-10-09

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录