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

相关文章

详解MySQL中DISTINCT去重的核心注意事项

《详解MySQL中DISTINCT去重的核心注意事项》为了实现查询不重复的数据,MySQL提供了DISTINCT关键字,它的主要作用就是对数据表中一个或多个字段重复的数据进行过滤,只返回其中的一条数据... 目录DISTINCT 六大注意事项1. 作用范围:所有 SELECT 字段2. NULL 值的特殊处

SpringBoot中使用Flux实现流式返回的方法小结

《SpringBoot中使用Flux实现流式返回的方法小结》文章介绍流式返回(StreamingResponse)在SpringBoot中通过Flux实现,优势包括提升用户体验、降低内存消耗、支持长连... 目录背景流式返回的核心概念与优势1. 提升用户体验2. 降低内存消耗3. 支持长连接与实时通信在Sp

MySQL 用户创建与授权最佳实践

《MySQL用户创建与授权最佳实践》在MySQL中,用户管理和权限控制是数据库安全的重要组成部分,下面详细介绍如何在MySQL中创建用户并授予适当的权限,感兴趣的朋友跟随小编一起看看吧... 目录mysql 用户创建与授权详解一、MySQL用户管理基础1. 用户账户组成2. 查看现有用户二、创建用户1. 基

MySQL 打开binlog日志的方法及注意事项

《MySQL打开binlog日志的方法及注意事项》本文给大家介绍MySQL打开binlog日志的方法及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录一、默认状态二、如何检查 binlog 状态三、如何开启 binlog3.1 临时开启(重启后失效)

SQL BETWEEN 语句的基本用法详解

《SQLBETWEEN语句的基本用法详解》SQLBETWEEN语句是一个用于在SQL查询中指定查询条件的重要工具,它允许用户指定一个范围,用于筛选符合特定条件的记录,本文将详细介绍BETWEEN语... 目录概述BETWEEN 语句的基本用法BETWEEN 语句的示例示例 1:查询年龄在 20 到 30 岁

MySQL DQL从入门到精通

《MySQLDQL从入门到精通》通过DQL,我们可以从数据库中检索出所需的数据,进行各种复杂的数据分析和处理,本文将深入探讨MySQLDQL的各个方面,帮助你全面掌握这一重要技能,感兴趣的朋友跟随小... 目录一、DQL 基础:SELECT 语句入门二、数据过滤:WHERE 子句的使用三、结果排序:ORDE

python使用库爬取m3u8文件的示例

《python使用库爬取m3u8文件的示例》本文主要介绍了python使用库爬取m3u8文件的示例,可以使用requests、m3u8、ffmpeg等库,实现获取、解析、下载视频片段并合并等步骤,具有... 目录一、准备工作二、获取m3u8文件内容三、解析m3u8文件四、下载视频片段五、合并视频片段六、错误

gitlab安装及邮箱配置和常用使用方式

《gitlab安装及邮箱配置和常用使用方式》:本文主要介绍gitlab安装及邮箱配置和常用使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装GitLab2.配置GitLab邮件服务3.GitLab的账号注册邮箱验证及其分组4.gitlab分支和标签的

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项

MySQL MCP 服务器安装配置最佳实践

《MySQLMCP服务器安装配置最佳实践》本文介绍MySQLMCP服务器的安装配置方法,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下... 目录mysql MCP 服务器安装配置指南简介功能特点安装方法数据库配置使用MCP Inspector进行调试开发指