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

embedded/2024/9/20 1:23:27/ 标签: go, 个人开发

文章目录

  • 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

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

1.1 学习过程

在这里插入图片描述

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

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

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

1.2.1 并发访问带来的问题

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

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

go">package mainimport ("fmt""sync"
)var counter int
var wg sync.WaitGroupfunc 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同时对该账户余额进行更新处理的时候,可能会导致余额错误

go">package mainimport ("fmt""sync"
)var balance int = 1000
var wg sync.WaitGroupfunc 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的操作。

go">package mainimport ("fmt""sync"
)var stock int = 10
var wg sync.WaitGroupfunc 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 接口:

go">type Locker interface {    Lock()    Unlock() 
} 

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

1、基本使用方法:

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

go">var mutex sync.Mutex 

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

go">mutex.Lock() 

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

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

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

go">package mainimport ("fmt""sync"
)var stock int = 10
var wg sync.WaitGroup
var mutex sync.Mutexfunc 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和临界区资源封装为一个类,这样更好的进行复用,不暴露内部实现

go">package mainimport ("fmt""sync"
)// StockManager 结构体封装了库存和互斥锁
type StockManager struct {stock intmutex 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.WaitGroupnumUsers := 100000wg.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.ppmy.cn/embedded/108354.html

相关文章

javascript中数组遍历的所有方法

作为后端程序员平常js用得少&#xff0c;但是数组遍历又是常用功能&#xff0c;遍历方法又有很多。在此记录一下&#xff0c;所有用得上的数组遍历方法。 1.for循环 最基本的数组遍历 特点: 通常配合数组的 .length 属性使用。索引从0开始&#xff0c;需要注意边界问题。 …

【数据分享】《中国城市统计年鉴》(1985-2023)全PDF版本 第一次补档

数据介绍 中国城市&#xff0c;如同一本生动的历史书&#xff0c;承载着经济、社会的快速变迁。《中国城市统计年鉴》记录了城市的发展轨迹&#xff0c;是我们理解城市化进程、洞察城市挑战的重要指南。 这份年鉴的数据庞大而详实&#xff0c;囊括了中国城市发展的多个方面。…

5.8 切换保护模式(5)

1 首先测试 了&#xff0c; 之前的代码 是 没有问题的&#xff0c;确实会 停在 汇编处。 1 首先是 设置 除了 CS 之外的寄存器 进入 32为模式 //为了使除了 cs 之外的 段选择寄存器也进入 32位模式。mov $16, %ax // 16为数据段选择子mov %ax, %dsmov %ax, %ssmov %ax, %esmov…

SpringMVC拦截器深度解析与实战

引言 Spring MVC作为Spring框架的核心模块之一&#xff0c;主要用于构建Web应用程序和RESTful服务。在Spring MVC中&#xff0c;拦截器&#xff08;Interceptor&#xff09;是一种强大的机制&#xff0c;它允许开发者在请求处理流程的特定点插入自定义代码&#xff0c;实现诸如…

今天又学到了——图编号关联章节号,QGIS下载文件存储的瓦片

记录教程来源&#xff1a;​​​​​​【Word图编号关联章节号】图片分章节 编号&#xff0c;图1-1、图2-1_哔哩哔哩_bilibili 上面链接这个实现的是这个效果&#xff1a; word自动目录及章节自动编号教程_哔哩哔哩_bilibili&#xff0c;这个的效果是自己设计多级列表&#xf…

uni-app全局引入js文件

js文件定义 对于js文件内方法的编写&#xff0c;可以采用以下两种形式&#xff0c;两种形式对应两种不同的文件引入&#xff1a; const showToast {test: function() {console.log("测试2")} } export default showToast 引入&#xff1a; import showToast from…

详解TensorRT的C++高性能部署以及C++部署Yolo实践

详解TensorRT的C高性能部署 一. ONNX1. ONNX的定位2. ONNX模型格式3. ONNX代码使用实例 二、TensorRT1 引言 三、C部署Yolo模型实例 一. ONNX 1. ONNX的定位 ONNX是一种中间文件格式&#xff0c;用于解决部署的硬件与不同的训练框架特定的模型格式的兼容性问题。 ONNX本身其…

PostgreSQL的基础知识

什么是数据库&#xff1f; 数据库&#xff08;Database&#xff09;是一个用于存储、检索、管理和分析数据的集合。它是按照一定的数据模型组织、存储的集合&#xff0c;具有统一的结构形式、定义的相互关系、制定的约束条件和一定的冗余度&#xff0c;以便于在各种用户、各种…

浅聊kubernetes RBAC

RBAC 基于角色&#xff08;Role&#xff09;的访问控制&#xff08;RBAC&#xff09;是一种基于组织中用户的角色来调节控制对计算机或网络资源的访问的方法。 RBAC 鉴权机制使用 rbac.authorization.k8s.io API 组来驱动鉴权决定&#xff0c; 允许你通过 Kubernetes API 动态…

C++学习笔记(8)

184、基于范围的 for 循环 对于一个有范围的集合来说&#xff0c;在程序代码中指定循环的范围有时候是多余的&#xff0c;还可能犯错误。 C11 中引入了基于范围的 for 循环。 语法&#xff1a; for (迭代的变量 : 迭代的范围) { // 循环体。 } 注意&#xff1a; 1&#xff09;迭…

实践reflex:项目架构解析

reflex 是一个使用纯Python构建全栈web应用的库&#xff0c;但是需要使用node&#xff0c;所以你懂的。 reflex安装&#xff1a;实践reflex&#xff1a;一个使用纯Python构建全栈web应用的库-CSDN博客 创建hello项目 (base) skyub:~$ mkdir hello (base) skyub:~$ cd hello/…

【Azure Redis】Redis-CLI连接Redis 6380端口始终遇见 I/O Error

问题描述 使用Redis-cli连接Redis服务&#xff0c;因为工具无法直接支持TLS 6380端口连接&#xff0c;所以需要使用 stunnel 配置TLS/SSL服务。根据文章(Linux VM使用6380端口(SSL方式)连接Azure Redis (redis-cli & stunnel) &#xff1a; https://www.cnblogs.com/luligh…

模型训练套路(二)

接模型训练套路&#xff08;一&#xff09;http://t.csdnimg.cn/gZ4Fm 得到预测的值&#xff1a;preds[1][1]&#xff0c; 输出目标&#xff1a;inputs target [0][1]&#xff1b; 查看两者的正确率&#xff0c;就看&#xff1a;predsinputs target 输出的结果&#xff1a…

前端WebSocket客户端实现

// 创建WebSocket连接 var socket new WebSocket(ws://your-spring-boot-server-url/websocket-endpoint);// 连接打开时触发 socket.addEventListener(open, function (event) {socket.send(JSON.stringify({type: JOIN, room: general})); });// 监听从服务器来的消息 socke…

K8S日志收集

本章主要讲解在 Kubernetes 集群中如何通过不同的技术栈收集容器的日志&#xff0c;包括程序直接输出到控制台日志、自定义文件日志等。 一、有哪些日志需要收集 为了更加方便的处理异常&#xff0c;日志的收集与分析极为重要&#xff0c;在学习日志收集之前&#xff0c;需要知…

QT基础 QPropertyAnimation简单学习

目录 1.简单介绍 2.使用步骤 3.部分代码示例 4.多项说明 5.信号反馈 6.自定义属性 1. 定义自定义属性 2. 使用 QPropertyAnimation 动画化自定义属性 3. 连接信号和槽 4.注意事项 7.更多高级示例 1.简单介绍 QPropertyAnimation是Qt中的一个类&#xff0c;用于实现属性…

idea安装并使用maven依赖分析插件:Maven Helper

在 IntelliJ IDEA 中安装并使用 Maven Helper 插件可以帮助你更方便地管理 Maven 项目的依赖&#xff0c;比如查看依赖树、排除冲突依赖等。以下是安装和使用 Maven Helper 插件的步骤&#xff1a; 安装 Maven Helper 插件 打开 IntelliJ IDEA 并进入你的项目。 在 IDE 的右下…

百度飞浆OCR半自动标注软件OCRLabel配置【详细

今天帮标注人员写了一份完整的百度飞浆OCR标注软件的安装配置说明书、以供标注人员使用 包括各种环境安装包一起分享出来【conda\python\label项目包、清华源配置文件、pycharm社区版安装包】 提取码&#xff1a;umys 1、解压并安装tools文件下的miniconda,建议安装在D盘下的…

Win32绕过UAC弹窗获取管理员权限

在早些年写一些桌面软件时&#xff0c;需要管理员权限&#xff0c;但是又不想UAC弹窗&#xff0c;所以一般是直接将UAC的级别拉到最低&#xff0c;或者直接禁用UAC的相关功能。 什么是UAC(User Account Control) 用户帐户控制 (UAC) 是一项 Windows 安全功能&#xff0c;旨在保…

Flink SQL 中常见的数据类型

Flink SQL 中常见的数据类型 目标 通过了解Flink SQL 中常见的数据类型,掌握正确编写Flink SQL 语句背景 Apache Flink 支持多种数据类型,这些数据类型被用于 Flink SQL 表达式、Table API 以及 DataStream API 中。以下是 Flink SQL 中常见的数据类型: 基本数据类型 Boo…