go语言并发编程-超详细mutex解析

news2024/9/21 10:42:42

文章目录

  • 1 go语言并发编程学习-mutex
    • 1.1 学习过程
    • 1.2 如何解决资源并发访问的问题?【基本用法】
      • 1.2.1 并发访问带来的问题
        • 1.2.1.1 导致问题的原因
      • 1.2.2 race detector检查data race
      • 1.2.3 mutex的基本实现机制以及使用方法
        • 1.2.3.1 具体使用-1
        • 1.2.3.1 具体使用-2

1 go语言并发编程学习-mutex

1.1 学习过程

在这里插入图片描述

1.2 如何解决资源并发访问的问题?【基本用法】

本小节主要为了解答以下问题:

  1. 为什么需要解决并发访问的问题?
  2. 怎么通过race detector来查找程序中的data race?
  3. mutex的基本机制和基本使用方法?

1.2.1 并发访问带来的问题

1. 多个goroutine并发更新计数器

在多个goroutine的情况下并发更新计数器,得到的值可能不符合预期。

package main

import (
    "fmt"
    "sync"
)

var counter int
var wg sync.WaitGroup

func increment() {
    defer wg.Done()
    counter += 100
}

func main() {
    wg.Add(1000) // 这个可以先不管,理解为,main函数需要等待goroutine都执行完才能退出,可以把wg相关全去掉,在main函数后面加上time.sleep(time.second * 10)
    for i := 0; i < 1000; i++ {
       go increment()
    }
    wg.Wait()
    fmt.Println("Final counter:", counter) // 期望输出 2,但可能输出 0 或 1
}

2. 更新用户的账户余额

在用户收入和支出的时候,如果不同的goroutine同时对该账户余额进行更新处理的时候,可能会导致余额错误

package main

import (
    "fmt"
    "sync"
)

var balance int = 1000
var wg sync.WaitGroup

func deposit(amount int) {
    defer wg.Done()
    balance += amount
}

func withdraw(amount int) {
    defer wg.Done()
    balance -= amount
}

func main() {
    wg.Add(2000)
    for i := 0; i < 1000; i++ {
       go deposit(1)
       go withdraw(1)
    }
    wg.Wait()
    fmt.Println("Final balance:", balance) // 期望输出 1000,但可能输出其他值
}

3. 秒杀系统

没有互斥锁的情况下,可能会出现超卖的情况。也就是商品已经没有了,但是还是可以进行出售商品,商品数量减1的操作。

package main

import (
    "fmt"
    "sync"
)

var stock int = 10
var wg sync.WaitGroup

func purchase() {
    defer wg.Done()
    if stock > 0 {
       fmt.Println("Stock:", stock)
       stock--
    }
}

func main() {
    wg.Add(100000)
    for i := 0; i < 100000; i++ {
       go purchase()
    }
    wg.Wait()
    fmt.Println("Final stock:", stock) // 期望输出 0,但可能输出负数
}

还有一些其他的场景,比如并发写入buffer等等,不解决并发访问的问题,就会发生很严重的后果。

1.2.1.1 导致问题的原因

并发访问问题的核心在于对共享资源的非原子性操作。临界区是指一段需要独占访问的代码块,多个goroutine在执行这段代码时,如果没有同步机制(如互斥锁)来保证互斥访问,就可能会产生数据竞争,导致数据不一致和其他问题。以下从临界区的角度来解释这些问题。下面分析多个goroutine并发更新计数器:

计数器的更新操作通常包括以下步骤:

  1. 读取当前计数器的值
  2. 对读取的值进行加法运算
  3. 将结果写回计数器

在并发情况下,如果两个goroutine同时执行这三个步骤中的任意一个步骤,没有同步机制来保证这三个步骤是原子操作,就会产生问题:

Goroutine 1: 读取 counter = 0
Goroutine 2: 读取 counter = 0
Goroutine 1: counter = 0 + 1 => counter = 1
Goroutine 2: counter = 0 + 1 => counter = 1 (覆盖了Goroutine 1的结果)

那么怎么在程序运行的时候发现呢?可以参考一下race detector工具

1.2.2 race detector检查data race

可以使用上文的秒杀系统作为例子。写这个的时候,图片转存失败,因此决定用极客上的图片。
1、 在执行go run counter.go的时候会出现以下结果,是可以正常运行通过的,但是结果不如愿:
在这里插入图片描述
2、但是假如race之后:go run -race main.go
在这里插入图片描述
这个警告不但会告诉你有并发问题,而且还会告诉你哪个goroutiine在哪一行对哪个变量有写操作,同时,哪个goroutine在哪一行对哪个变量有读操作,就是这些并发的读写访问,引起了datarace。
例子中的goroutine 10 对内存地址0x000126010有读的操作(ctounter.go文件第16行),同时,goroutine7对内存地址0x00c000126010有写的操作(counter.go文件第16行)。而且还可能有多个goroutine在同时进行读写,所以,警告信息可能会很长。

总结一下,通过在编译的时候插入一些指令,在运行时通过这些插入的指令检测并发读写从而发现data race问题,就是这个工具的实现机制。
既然这个例子存在data race问题,我们就要想办法来解决它。这个时候,我们这节课的主角Mutex就要登场了,它可以轻松地消除掉data race。
具体怎么做呢?下面,我就结合这个例子,来具体说一说Mutex的基本用法。

1.2.3 mutex的基本实现机制以及使用方法

在这里插入图片描述
​ mutex的基本实现的机制,就是每次只允许一个goroutine进入临界区,具体就是进入临界区的时候给临界区加上一个锁,禁止其他goroutine进入临界区,在退出临界区的时候释放锁,从而允许其他goroutine进入。

Mutex 是 Go 语言中常用的同步原语,用于控制对共享资源的独占访问。Mutex 实现了 sync.Locker 接口,该接口定义了两个方法:LockUnlock。在解释 Mutex 的基本使用方法之前,先简单介绍一下 sync.Locker 接口:

type Locker interface {    
	Lock()    
	Unlock() 
} 

任何实现了 LockUnlock 方法的类型,都可以被视为 Locker,所以 sync.Mutex 也实现了这个接口。以下是一个基本的 sync.Mutex 的使用方法:

1、基本使用方法:

声明一个sync.Mutex类型的变量,无需初始化

var mutex sync.Mutex 

2、在需要保护的临界区前调用 Lock 方法:

mutex.Lock() 

3、在临界区结束后调用 Unlock 方法:

mutex.Unlock() 
1.2.3.1 具体使用-1

这种使用,主要是在临界区代码中直接使用mutex即可。

package main

import (
    "fmt"
    "sync"
)

var stock int = 10
var wg sync.WaitGroup
var mutex sync.Mutex

func purchase() {
    defer wg.Done()
    mutex.Lock()         // 加锁,进入临界区
    defer mutex.Unlock() // 确保解锁
    if stock > 0 {
       fmt.Println("Stock:", stock)
       stock--
    }
}

func main() {
    wg.Add(100000)
    for i := 0; i < 100000; i++ {
       go purchase()
    }
    wg.Wait()
    fmt.Println("Final stock:", stock) // 期望输出 0,但可能输出负数
}
1.2.3.1 具体使用-2

该使用是把mutex和临界区资源封装为一个类,这样更好的进行复用,不暴露内部实现

package main

import (
    "fmt"
    "sync"
)

// StockManager 结构体封装了库存和互斥锁
type StockManager struct {
    stock int
    mutex sync.Mutex
}

// NewStockManager 创建一个新的 StockManager
func NewStockManager(initialStock int) *StockManager {
    return &StockManager{stock: initialStock}
}

// Purchase 尝试购买一个商品
func (sm *StockManager) Purchase() bool {
    sm.mutex.Lock()
    defer sm.mutex.Unlock()

    if sm.stock > 0 {
       sm.stock--
       fmt.Println("Purchase successful, remaining stock:", sm.stock)
       return true
    }
    fmt.Println("Purchase failed, out of stock")
    return false
}

// GetStock 获取当前库存
func (sm *StockManager) GetStock() int {
    sm.mutex.Lock()
    defer sm.mutex.Unlock()
    return sm.stock
}

func main() {
    sm := NewStockManager(10)
    var wg sync.WaitGroup

    numUsers := 100000
    wg.Add(numUsers)
    for i := 0; i < numUsers; i++ {
       go func() {
          defer wg.Done()
          sm.Purchase()
       }()
    }
    wg.Wait()
    fmt.Println("Final stock:", sm.GetStock())
}

下一篇:mutex的原理以及常见的错误。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2099102.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

@Tanstack/vue-query 的使用介绍

Tanstack/vue-query 的使用介绍 前言 在今年的vue conf 会议上&#xff0c;提到了vue-query这个库&#xff0c;这里对它的基本使用做一个介绍。 会议资料地址&#xff1a; https://vueconf.cn/ Tanstack-query的前身是react-query&#xff0c;是一个本地的服务端状态管理的库…

deepin 23 下如何运行绝大数 Windows 游戏?

查看原文 最近有很多 deepiner 一直在询问&#xff1a; “deepin 真的可以玩游戏&#xff1f;” “如何在 deepin 上玩游戏&#xff1f;能玩 windows 上的游戏吗&#xff1f;” 答案是肯定的 “必须能&#xff01;”&#xff0c;下文和大家分享一下在 deepin 上玩游戏实现的…

使用IoC容器--Ninject

Ninject Ninject是一个较新的开源的IoC容器。这是简单和可扩展的。你可以从下面的位置下载IoC容器。 Ninject 或者您可以使用 NuGet 向您的项目添加Ninject。让我们从NuGet向我们的项目中添加Ninject。只需转到您的项目引用并右键单击&#xff0c;然后 ManageNuGet Packages&a…

Java基础 2. Java基础语法

Java基础 2. Java基础语法 文章目录 Java基础 2. Java基础语法2.1. 标识符2.1.1. 标识符的命名规则 :2.1.2. 标识符的命名规范: 2.2. 关键字2.3. 字面量2.3.1. Java中有哪些字面量2.3.2. 加号运算符 2.4. 变量2.5. 二进制2.6. 八进制与十六进制2.7. 原码反码补码2.7.1. byte与b…

每日OJ_牛客_字符串计数(模拟26进制)

目录 牛客_分解因数&#xff08;简单模拟&#xff09; 解析代码 牛客_分解因数&#xff08;简单模拟&#xff09; 字符串计数_美团笔试题_牛客网 解析代码 题目意思&#xff1a;按照字典序列&#xff1a;找到s1和s2之间长度在len1和len2范围内的字符串个数。直接做不好处理&…

UE4 使用AndroidGameDevelopmentExtension(AGDE)对安卓客户端做“断点调试”与“代码热更”

本文的目的 主要介绍了如何通过AndroidGameDevelopmentExtension工具、Visual Studio 2022来进行安卓包调试。 流程全过程 1、安装JDK、Gradle、Android NDK、Android SDK等环境如下&#xff0c;请自行前往官网下载&#xff0c;并确认环境参数配置正确。 2、安装 AndroidGame…

黑悟空上线即登顶,如何一路火遍全网?哪些营销方式值得关注?

几乎人人都有一个大圣梦&#xff0c;就像每个品牌也都想成为“黑神话&#xff1a;悟空”&#xff08;以下简称“黑悟空”&#xff09;。 作为国内首款3A游戏&#xff0c;黑悟空上线即登顶&#xff0c;直接刷爆全网。虽然目前热度渐歇&#xff0c;但创造的流量神话仍被津津乐道。…

大佬借助ChatGPT写论文发刊到手软,四个步骤20个顶级学术提示词指令

大家好,感谢关注。我是七哥,一个在高校里不务正业,折腾学术科研AI实操的学术人。关于使用ChatGPT等AI学术科研的相关问题可以和作者七哥(yida985)交流,多多交流,相互成就,共同进步,为大家带来最酷最有效的智能AI学术科研写作攻略。经过数月爆肝,终于完成学术AI使用教…

Mybatis【分页插件,缓存,一级缓存,二级缓存,常见缓存面试题】

文章目录 MyBatis缓存分页延迟加载和立即加载什么是立即加载&#xff1f;什么是延迟加载&#xff1f;延迟加载/懒加载的配置 缓存什么是缓存&#xff1f;缓存的术语什么是MyBatis 缓存&#xff1f;缓存的适用性缓存的分类一级缓存引入案例一级缓存的配置一级缓存的工作流程一级…

《OpenCV计算机视觉》—— 图像形态学(腐蚀、膨胀等)

文章目录 一、图像形态学基本概念二、基本运算1.简单介绍2.代码实现 三、高级运算1.简单介绍2.代码实现 一、图像形态学基本概念 图像形态学是图像处理科学的一个独立分支&#xff0c;它基于集合论和数学形态学的理论&#xff0c;专门用于分析和处理图像中的形状和结构。图像形…

分贝通助力元气森林企业支出一体化降本提效

凭借着“0糖0脂0卡”这句广告语,元气森林几乎是一锤砸中了年轻消费者的内心,让“好喝不胖”深入人心,成为了国内饮品消费的新风向标。如果我们从近两年的快消饮品中选出几款深受消费者喜爱的“国货品牌”的话,相信「元气森林」一定上榜。 元气森林成立于2016年,旗下拥有元气森林…

深入理解并实现——快排【C语言版】

目录 一、快排介绍及其思想 二、hoare版本 三、前后指针版 四、挖坑法 五、优化版本 5.1 三数取中 5.2 小区间优化 六 、非递归实现快排 七、三路划分 八、introsort 小结 一、快排介绍及其思想 快速排序是C.R.A.Hoare于1962年提出的一种划分交换排序。它采用了一…

【数据结构】Map的使用与注意事项

文章目录 概念模型Map 的使用put() 和 get()getOrDefault()remove()keySet()entrySet() 注意事项 概念 Map 和 set 是一种专门用来进行搜索的容器或者数据结构&#xff0c;其搜索的效率与其具体的实例化子类有关。 以前常见的搜索方式有&#xff1a; 直接遍历&#xff0c;时间…

Spring AOP(下)原理

本文我们来学习 Spring AOP 的原理&#xff0c;也就是 Spring 是如何实现 AOP 的。Spring AOP 是基于动态代理来实现 AOP 的&#xff1b; 1. 代理模式 1.1 代理弄模式的定义 代理模式&#xff0c;也叫委托模式。 定义&#xff1a;为其他对象提供一种代理以控制这个对象的访问…

【第三版 系统集成项目管理工程师】第14章 收尾过程组

持续更新。。。。。。。。。。。。。。。 【第三版】第十四章 收尾过程组 14.1结束项目或阶段14.1.1主要输入1.项目章程-P5392.项目管理计划-P5393.项目文件-P5394.验收的可交付成果-P5405.协议-P5406.采购文档-P540 14.1.2主要输出1.最终产品、服务或成果-P5402.项目最终报告…

如何在IntelliJ IDEA中将Tab设置为4个空格

前言 IntelliJ IDEA是一个强大的开发工具&#xff0c;支持多种编程语言。为了保持代码整洁一致&#xff0c;开发者经常需要调整编辑器中的Tab和缩进设置。 步骤1: 打开设置 首先&#xff0c;启动IntelliJ IDEA。在主界面上方的菜单栏中找到 File&#xff08;文件&#xff09…

体育直播平台开发:初创公司突破资金与市场的双重挑战

在当今竞争激烈的数字娱乐行业中&#xff0c;体育直播平台的发展潜力巨大。然而&#xff0c;面对如此巨大的蓝海市场&#xff0c;对于初创公司或中小型平台而言&#xff0c;高昂的开发和运营成本可能成为进入该行业的主要障碍。本文将探讨一些可行的低成本开发策略&#xff0c;…

Matlab三维图的坐标轴标签 自动平行坐标/自动旋转

下载解压工具包&#xff1a; https://www.mathworks.com/matlabcentral/fileexchange/49542-phymhan-matlab-axis-label-alignment 添加至MATLAB路径: 在三维绘图后增加下列语句即可 ax struct(Axes, gca); align_axislabel([],ax) h3d rotate3d; set(h3d,ActionPreCa…

k8s项目的发布(金丝雀发布)

目录 三种发布方式 1.蓝绿发布 2.金丝雀发布&#xff08;灰度发布&#xff09; 实验&#xff1a;k8s实现金丝雀发布 3.滚动发布&#xff08;默认形式&#xff09; 因为应用升级以及新旧业务切换&#xff0c;所以在这个过程当中如何保证对外的服务正常是一个非常重要的问题…

/单元测试

承接上文 统一异常处理&#xff0c;封装结果-CSDN博客 ******************************************** 登录业务 Service public class EmployeeServiceImpl extends ServiceImpl<EmployeeMapper, Employee> implements EmployeeService {Resourceprivate JwtUtils j…