浅谈go语言中的读写锁和互斥锁

Hello,各位小伙伴大家好,我是小栈君,近期气温有所下降,希望各位小伙伴记得防寒保暖,不要感冒了哦。

本期分享主题是关于go语言中的锁的应用场景,以及为各位小伙伴介绍实战应用中最为广泛的读写锁和互斥锁。

互联网生态的日益繁荣,人们的生活便利得到了极大的提高,通过网上操作我们基本上可以实现很多需求。

网站疯狂访问的背后应对的是一波接一波的挑战。所以在应对系统的稳定和并发的时候,程序中的“锁”就孕育而生。

互斥锁

在编程中,引入了对象互斥锁的概念,来保证共享数据操作的完整性。每个对象都对应于一个可称为" 互斥锁" 的标记,这个标记用来保证在任一时刻,只能有一个线程访问该对象。

也就是将共享资源变成独占资源。互斥锁的应用场景通常是写大于读操作的,它不同于读写锁的读者随意访问,而写者只有一个。

它代表的资源就是一个,不管是读者还是写者,只要谁拥有了它,那么其他人就只有等待解锁后,隐约在脑海中浮现出“宝刀屠龙,谁与针锋”的话语。

其实我们可以很形象的理解一下互斥锁,资源就好比是一个厕所,很多人都想上厕所,但是坑位只有一个,那么谁获取了互斥锁,那么谁就有权利进去,其他人只有在门口排队等待。也就是我们通常所说的阻塞。

图片1.png

在go语言的sync包中也是有对于互斥锁的解释,互斥锁的结构体很简单,并且他的接口就只有一个加锁和一个解锁操作。

当value为空时就是一个解锁的互斥锁,也就是其他人都可以来使用。并且当互斥锁第一次使用的时候就不能再被复制。

图片2.png

并且在代码中也有很详细的说明,有兴趣的小伙伴可以参考代码源码进行了解。大致的意思就是说互斥锁有两种模式,正常和饥饿模式。

在正常的模式下采用的是FIFO模式即先进先出,但是会被等待者唤醒。没有拥有互斥锁的等待者会同新来的协程进行竞争,获取锁的使用权。

但是新来的协程拥有一个优势就是他们是运行在CPU上的,而之前有可能会有很多进程或协程需要被唤醒,所以他们有可能在毫秒之间就被人插队了。

也就是新来的不需要被唤醒直接获取到。如果一个在1ms的时间内没有获取到互斥量,那么它将进入到饥饿模式。

也就是说在互斥锁的饥饿模式下他会进行有序的交接。也就是会将互斥锁的所有权进行移交到排在前面的等待者。

而新来的等待者想要获取互斥锁就只有乖乖排队。他也不会试图去抢占互斥锁。如果说在饥饿模式下他是最后一个互斥锁的拥有者的话,或是等待少于1ms获取锁,那么他就会重新转变为正常模式。

其实在正常模式下协程会有更好的性能,但是饥饿模式是为了预防更多不确定的情况。

互斥锁实战:

我们先进行一个简单的模拟,定义个Map进行模拟数据库对象,制定一个Map的切片来做为数据对象的id和name,并且进行数据的初始化,当我们开启10个并发请求进行修改某一个值的时候。

图片3.png

我们可以看到最终的结果是修改成功了。但是各位小伙伴我们是真的就没有任何问题了么?

在go语言中我们其实可以检查是否有问题可以使用 go build -race 来进行数据竞争检测。

[小知识:以后小伙伴在不确定的情况下都可以进行使用命令进行验证,当我们不确定打包工具有哪些命令的时候,我们可以用go help build 来进行查看]

4.jpg

双击后我们可以看到出现了以下问题,看来当初我们直接使用goland进行运行看到的表现往往有一丝丝的不妥当,还是太年轻了啊!

5.jpg

我们也可以看到在点击文件后得到的结论是确实对于共享资源的抢占是会导致问题的,而且在go语言中我们也是很友好的进行了提前的测试,避免了在线上问题排查。

对于线程相关的问题在生产上其实排查相对而言是有点困难的。在文件中也可以明确的显示出问题出现在17行,也就是我们在对于Map切片进行操作的时候。

6.jpg

所以我们针对多并发访问的时候需要对共享资源进行加锁处理,目前模拟的应用场景是写大于读的时候。我们使用互斥锁进行相应的操作。

得到的结论和之前的一样,但是我们同样需要对这段程序进行go build -race 操作。

7.jpg

最终得到了正确无误的操作。以上就是我们关于互斥锁的初步使用。按照使用规范来讲,不管是互斥锁和读写锁,我们都应该尽可能的对于小范围的进行使用,在关键处进行使用,避免程序拥有大量的阻塞。

读写锁

读写锁实际是一种特殊的自旋锁,它把对共享资源的访问者划分成读者和写者,读者只对共享资源进行读访问,写者则需要对共享资源进行写操作。

这种锁相对于自旋锁而言,能提高并发性,因为在多处理器系统中,它允许同时有多个读者来访问共享资源,最大可能的读者数为实际的逻辑CPU数。

写者是排他性的,一个读写锁同时只能有一个写者或多个读者(与CPU数相关),但不能同时既有读者又有写者。在读写锁保持期间也是抢占失效的。

如果读写锁当前没有读者,也没有写者,那么写者可以立刻获得读写锁,否则它必须自旋在那里,直到没有任何写者或读者。

如果读写锁没有写者,那么读者可以立即获得该读写锁,否则读者必须自旋在那里,直到写者释放该读写锁。

所以针对于系统中需要读多写少的情况下,我们就需要使用到读写锁进行应对程序的并发,保护程序的安全、稳定、高效的运行。

在go语言中内置包中已经实现了关于读写锁操作,我们在系统中需要使用可以很方便的进行操作。

8.jpg

在代码注释中很明确的可以看出,在go语言中的sync包中的RWMutex就是一个读写互斥锁,该锁可以有任意数量的读者或是只有一个写者持有,形象的说就好比一场电影可以有无数的人来看,但是导演就只有一个。

当值为空的时候,他是处于解锁状态,并且在读写锁第一次使用的时候是不允许被拷贝的。如果一个协程(goroutine)持有读的权限,另一个协程进行锁操作,那么没有任何一个协程能够再持有读锁,除非被释放。

当然开发者也提醒了禁止进行递归读锁定,是为了保证锁能够一直可用。大概意思就是说这部电影不能让你一个人独自看,毕竟独乐乐不如众乐乐嘛。

在代码中我们常用的两个操作就是Lock 和Unlock 即加解锁操作,通过代码中我们可以知道go语言最多读者可以高达10亿个,已经能够完美的满足到我们的业务需求了。

9.jpg

读写锁实战:

接下来我们将模拟一下关于网页中的读写锁操作,读写锁的应用场景多数是类似于朋友圈或微博的并发场景下的读大于写的场景,所以我们用读写两个循环协程进行模拟数据库的请求操作,并是使用count进行原子计数,最终得到的结果如下:

10.jpg

当然如果我们依旧关闭锁操作会得到相同的结果么?

11.jpg

使用goland执行后的结果和之前加锁的状态是是没有什么区别,但是真的没有什么区别么?

让我们使用一下go build -race 命令,最后得到的结果如下:

12.jpg

表面风轻云淡的情况下内部已经翻江倒海了,所以小栈君在这里也是奉劝各位三思而后行。

读写锁与互斥锁性能大比拼

读写锁和互斥锁的各自实战情况已经初略的给各位分享了一遍,总体而言用法是比较简单的,并且有兴趣的小伙伴可以看看go语言的内置包,后续我也会陆续为大家分享关于go语言实现二叉树,链表结构等文章,让大家更加深入的感受到go语言的魅力。

当然我也在筹划其他语言的分享,所以各位小伙伴,如果你喜欢我的文章,麻烦分享并关注小栈君哦。

回归正题,我们在使用各自的使用场景下并没有感受到读写锁鱼互斥锁性能上有多大的区别。所以小栈君接下来就一个场景分别使用两个锁来进行模拟请求计数,得出结论。

我们首先使用读写锁进行模拟如图所示:

13.jpg

我们先定义读写锁、数据库数据、还有相关的原子计数器。这里有一个小知识点,我们在计数的时候用了atomic,这样可以保证计数的准确和安全。然后开启了一个写协程和一百个读协程。

为了加大请求力度我们有在每个协程里开启了无线循环读取,用1毫秒模拟查询时间,用10毫秒模拟写时间。最终得出的结论是计数器计算到了16万接近17万次。

14.jpg

然后相同的代码我们仅仅只修改了一下锁的机制,将读写锁改成了互斥锁,各位可以看到效果就是代码执行了1695次,相差是非常大的。

15.jpg
16.jpg

以上就是关于go语言中的互斥锁和读写锁的分享。事实证明,只有锁用的对,我们就可以早些下班啦。

好了,今天的分享就到这啦,如果你喜欢我的分享,麻烦你点击一个好看或赞,我是小栈君,不定期分享IT干货,包括但不限于区块链、大数据、Python、go、等系列专题。原创不易,更新较慢,多多包涵。希望与你共同成长。我们下期再见啦,拜了个拜~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容