mysql 悲观锁使用

2024-08-25 18:28
文章标签 mysql 使用 database 悲观

本文主要是介绍mysql 悲观锁使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

悲观锁是一种数据库锁定机制,它假设每次操作都会发生并发冲突,因此在执行任何需要读取或写入数据的操作之前,先获取锁,防止其他事务对该数据进行修改。悲观锁确保了操作的独占性,以防止数据被其他事务同时修改,从而保证数据的一致性。

悲观锁分为两种:
1、共享锁(S锁):允许多个事务同时读取数据,但不能修改数据。
2、排他锁(X锁):禁止其他事务读取或修改数据,只有获取锁的事务可以操作数据。
使用场景:适用于并发度较高的场景,尤其是在可能发生数据冲突的情况下。
MySQL中,悲观锁通常是通过 SELECT ... FOR UPDATESELECT ... LOCK IN SHARE MODE 来实现的

1、 排他锁(Exclusive Lock, X锁)

场景:你想读取某条记录并在后续操作中修改它,确保在此期间没有其他事务可以读取或修改这条记录。比如:银行转账、库存管理、订单系统中的竞价 、避免重复处理的任务调度

一、银行转账

在银行系统中,确保用户账户余额的正确性至关重要。例如,进行转账时需要确保读取的余额在更新之前不会被其他事务修改。
假设我们有一个 accounts 表,存储用户的账户信息,包括 id、balance 等字段。用户从一个账户转账到另一个账户。

START TRANSACTION;-- 查询并锁定源账户的余额
SELECT balance FROM accounts WHERE id = 1 FOR UPDATE;-- 确保余额足够
-- (在应用程序中检查)
-- 假设 balance > 100-- 执行扣款操作
UPDATE accounts SET balance = balance - 100 WHERE id = 1;-- 查询并锁定目标账户的余额
SELECT balance FROM accounts WHERE id = 2 FOR UPDATE;-- 执行存款操作
UPDATE accounts SET balance = balance + 100 WHERE id = 2;COMMIT;

FOR UPDATE 确保在读取源账户的余额后,该余额不会被其他事务修改,避免了在扣款操作中可能发生的并发问题。

二、 库存管理

在库存管理系统中,确保在查询库存和更新库存数量之间没有其他事务干扰,避免超卖或库存不足的情况。

场景:
一个电子商务网站在处理用户订单时,需要确保产品库存足够并且在处理订单时不被其他订单处理影响。

START TRANSACTION;-- 查询并锁定产品库存
SELECT stock FROM products WHERE product_id = 101 FOR UPDATE;-- 确保库存足够
-- (在应用程序中检查)
-- 假设 stock >= 1-- 执行减库存操作
UPDATE products SET stock = stock - 1 WHERE product_id = 101;COMMIT;

FOR UPDATE 确保在读取产品库存后,库存数据在减库存之前不会被其他事务改变,从而避免库存数据的不一致问题。

三、订单系统中的竞价

在竞价系统中,多个用户可能会同时对同一商品进行竞价。需要确保在读取当前最高竞价后进行更新时,竞价数据不会被其他竞价操作影响。

场景:

一个在线拍卖系统允许多个用户同时对一个商品进行竞价。系统需要确保在更新最高竞价时没有其他用户同时进行操作。

START TRANSACTION;-- 查询并锁定当前最高出价
SELECT max_bid FROM bids WHERE item_id = 123 FOR UPDATE;-- 确保当前出价超过了最高出价
-- (在应用程序中检查)
-- 假设 max_bid < 500-- 更新最高出价
UPDATE bids SET max_bid = 500, bidder_id = 10 WHERE item_id = 123;COMMIT;

FOR UPDATE 确保在读取当前最高出价后,没有其他竞价操作在此时更新该数据,避免了竞价结果的不一致性。

四、 避免重复处理的任务调度

在任务调度系统中,可能需要从任务表中选择一个待处理的任务并将其标记为已处理,避免多个调度器同时处理同一个任务。
有一个 tasks 表,存储了多个待处理的任务,多个调度器(或工作线程)从中获取任务来处理。

START TRANSACTION;-- 查询并锁定一个待处理任务
SELECT id FROM tasks WHERE status = 'pending' LIMIT 1 FOR UPDATE;-- 更新任务状态为处理中
UPDATE tasks SET status = 'processing' WHERE id = 1;COMMIT;

FOR UPDATE 确保在查询到待处理任务后,没有其他调度器在此时获取同一个任务并进行处理,避免了任务重复处理的情况。

2、共享锁(Shared Lock, S锁)

一、只读操作中的一致性保障

场景:

在报表生成或分析过程中,需要从多个表中读取数据,确保在读取期间数据不会被其他事务修改,以防止数据不一致。

START TRANSACTION;-- 对数据加共享锁,确保读取期间数据不被修改
SELECT balance FROM accounts WHERE id = 1 LOCK IN SHARE MODE;-- 执行其他读取操作或分析
SELECT * FROM transactions WHERE account_id = 1 LOCK IN SHARE MODE;COMMIT;

LOCK IN SHARE MODE 会对查询的数据行加共享锁,其他事务可以继续读取这些行,但不能修改这些行(不能加排他锁)。
适用于读取操作后不需要立即修改数据的场景,例如报表生成。

二、 避免读写冲突

有一个系统需要检查用户是否有权限访问某些资源。在权限检查过程中,必须确保权限数据不会被其他事务修改。

START TRANSACTION;-- 对权限数据加共享锁,确保读取期间权限数据不被修改
SELECT permission_level FROM user_permissions WHERE user_id = 42 LOCK IN SHARE MODE;-- 根据权限数据执行相应操作
-- 在应用程序中进行进一步的权限验证COMMIT;

共享锁确保了在检查用户权限时,权限数据不会被其他事务修改,这样可以避免权限验证时的数据不一致问题。

三、 复杂查询中的数据一致性

在执行一个复杂的查询操作时,需要确保在查询的过程中,数据不会被其他事务修改。例如,统计某一段时间内的订单总数和金额。

START TRANSACTION;-- 对订单数据加共享锁,确保读取期间订单数据不被修改
SELECT COUNT(*) AS total_orders, SUM(amount) AS total_amount 
FROM orders 
WHERE order_date BETWEEN '2024-01-01' AND '2024-01-31' 
LOCK IN SHARE MODE;COMMIT;

在统计订单数据时,使用共享锁可以防止在查询过程中订单数据被其他事务修改,确保统计结果的准确性。

四、防止“脏读”的数据校验

在某些需要多次读取同一数据进行校验的场景下,可以使用共享锁来确保校验期间数据不会被其他事务修改,避免“脏读”。

START TRANSACTION;-- 第一次读取
SELECT stock FROM products WHERE product_id = 101 LOCK IN SHARE MODE;-- 假设在应用程序中有某些校验逻辑
-- 如果校验通过,再次读取SELECT stock FROM products WHERE product_id = 101 LOCK IN SHARE MODE;COMMIT;

共享锁确保在校验过程中的两次读取之间,数据不会被修改,避免了在两次读取之间发生“脏读”而导致的校验失败。

这篇关于mysql 悲观锁使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1106316

相关文章

如何使用celery进行异步处理和定时任务(django)

《如何使用celery进行异步处理和定时任务(django)》文章介绍了Celery的基本概念、安装方法、如何使用Celery进行异步任务处理以及如何设置定时任务,通过Celery,可以在Web应用中... 目录一、celery的作用二、安装celery三、使用celery 异步执行任务四、使用celery

使用Python绘制蛇年春节祝福艺术图

《使用Python绘制蛇年春节祝福艺术图》:本文主要介绍如何使用Python的Matplotlib库绘制一幅富有创意的“蛇年有福”艺术图,这幅图结合了数字,蛇形,花朵等装饰,需要的可以参考下... 目录1. 绘图的基本概念2. 准备工作3. 实现代码解析3.1 设置绘图画布3.2 绘制数字“2025”3.3

Jsoncpp的安装与使用方式

《Jsoncpp的安装与使用方式》JsonCpp是一个用于解析和生成JSON数据的C++库,它支持解析JSON文件或字符串到C++对象,以及将C++对象序列化回JSON格式,安装JsonCpp可以通过... 目录安装jsoncppJsoncpp的使用Value类构造函数检测保存的数据类型提取数据对json数

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

springboot整合 xxl-job及使用步骤

《springboot整合xxl-job及使用步骤》XXL-JOB是一个分布式任务调度平台,用于解决分布式系统中的任务调度和管理问题,文章详细介绍了XXL-JOB的架构,包括调度中心、执行器和Web... 目录一、xxl-job是什么二、使用步骤1. 下载并运行管理端代码2. 访问管理页面,确认是否启动成功

Mysql 中的多表连接和连接类型详解

《Mysql中的多表连接和连接类型详解》这篇文章详细介绍了MySQL中的多表连接及其各种类型,包括内连接、左连接、右连接、全外连接、自连接和交叉连接,通过这些连接方式,可以将分散在不同表中的相关数据... 目录什么是多表连接?1. 内连接(INNER JOIN)2. 左连接(LEFT JOIN 或 LEFT

使用Nginx来共享文件的详细教程

《使用Nginx来共享文件的详细教程》有时我们想共享电脑上的某些文件,一个比较方便的做法是,开一个HTTP服务,指向文件所在的目录,这次我们用nginx来实现这个需求,本文将通过代码示例一步步教你使用... 在本教程中,我们将向您展示如何使用开源 Web 服务器 Nginx 设置文件共享服务器步骤 0 —

Java中switch-case结构的使用方法举例详解

《Java中switch-case结构的使用方法举例详解》:本文主要介绍Java中switch-case结构使用的相关资料,switch-case结构是Java中处理多个分支条件的一种有效方式,它... 目录前言一、switch-case结构的基本语法二、使用示例三、注意事项四、总结前言对于Java初学者

Golang使用minio替代文件系统的实战教程

《Golang使用minio替代文件系统的实战教程》本文讨论项目开发中直接文件系统的限制或不足,接着介绍Minio对象存储的优势,同时给出Golang的实际示例代码,包括初始化客户端、读取minio对... 目录文件系统 vs Minio文件系统不足:对象存储:miniogolang连接Minio配置Min