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

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)