MySQL锁三部曲:临键、间隙与记录的奇妙旅程

欢迎来到我的博客,代码的世界里,每一行都是一个故事


在这里插入图片描述

MySQL锁三部曲:临键、间隙与记录的奇妙旅程

    • 前言
    • 临键锁的奥秘
    • 间隙锁
    • 记录锁

前言

在数据库世界中,锁是维护数据完整性的一种关键机制。而MySQL中的临键锁、间隙锁和记录锁则是锁定数据的三大法宝。本文将引领读者进入这场锁的盛宴,深刻理解这三种锁的独特作用,以及如何在实际应用中灵活运用它们。

临键锁的奥秘

临键锁(Next-Key Locks)是很独特的一种锁,直观上来说可以看做是一个记录锁和间隙锁的组合。也就是说临键锁不仅仅是会用记录锁锁住命中的记录,也会用间隙锁锁住记录之间的空隙。临键锁和数据库隔离级别的联系最为紧密,它可以解决在可重复读隔离级别之下的幻读问题。间隙锁是左开右开,而临键锁是左开右闭。在数据库中,“临键锁"通常指的是"临键锁定”(Row-level lock),这是一种锁定记录的机制,确保对特定记录的独占访问。以下是临键锁的基本概念以及在数据库中如何使用它来确保对特定记录的独占访问:

基本概念:

  1. 行级锁: 临键锁是行级锁的一种,它锁定表中的特定行而不是整个表。

  2. 锁粒度: 行级锁允许并发事务在表中的不同行上工作,从而提高系统的并发性。

  3. 锁的状态: 临键锁可以处于不同的状态,包括共享锁(Shared Lock)和独占锁(Exclusive Lock)。

  4. 共享锁和独占锁:

    • 共享锁: 允许多个事务同时获取锁,用于读取操作,表示事务不会修改数据。
    • 独占锁: 只允许一个事务获取锁,用于写入操作,表示事务可能修改数据。

在数据库中如何使用临键锁:

  1. SELECT语句中的共享锁:

    • 当事务执行SELECT语句时,可以使用共享锁来确保其他事务不会在相同的记录上执行写操作。
    • 通过使用SELECT … FOR SHARE语法,事务可以获取共享锁。
    SELECT * FROM your_table WHERE your_condition FOR SHARE;
    
  2. UPDATE和DELETE语句中的独占锁:

    • 当事务执行UPDATE或DELETE语句时,可以使用独占锁来确保其他事务不会同时修改或删除相同的记录。
    • 通过使用UPDATE或DELETE语句时的FOR UPDATE语法,事务可以获取独占锁。
    UPDATE your_table SET your_column = 'new_value' WHERE your_condition FOR UPDATE;
    
  3. INSERT语句中的独占锁:

    • 当事务执行INSERT语句时,可以使用独占锁来确保其他事务不会同时在相同的记录位置插入数据。
    • 通过使用INSERT … ON DUPLICATE KEY UPDATE或INSERT IGNORE语句时的FOR UPDATE语法,事务可以获取独占锁。
    INSERT INTO your_table (your_columns) VALUES (your_values) ON DUPLICATE KEY UPDATE your_column = 'new_value' FOR UPDATE;
    

通过合理使用临键锁,可以在多个并发事务中确保对数据库表中特定记录的独占访问,从而维护数据的一致性和完整性。需要注意的是,过度使用锁可能导致性能问题,因此在设计和优化时需要权衡并考虑具体的业务场景。

间隙锁

间隙锁的作用:

间隙锁(Gap Lock)是一种在数据库中用于锁定一个范围而不是单个记录的锁。其作用在于:

  1. 确保范围内没有新数据插入: 通过使用间隙锁,可以确保在一个范围内没有新的记录被插入,从而避免并发事务在同一个范围内插入新的数据。

  2. 防止幻读: 间隙锁也可以防止幻读,即在同一个范围内确保其他事务不会插入新的记录,防止当前事务读到其他事务插入的未提交数据。

在并发操作中如何使用间隙锁:

考虑以下情境,使用间隙锁来避免不可预知的问题:

  1. 事务1:

    START TRANSACTION;
    SELECT * FROM your_table WHERE your_column BETWEEN 10 AND 20 FOR UPDATE;
    
  2. 事务2:

    START TRANSACTION;
    -- 此时间隙锁会锁定范围 [10, 20],防止其他事务插入新数据
    INSERT INTO your_table (your_column) VALUES (15);
    COMMIT;
    
  3. 事务1:

    -- 在此时,事务1再次执行相同的查询
    SELECT * FROM your_table WHERE your_column BETWEEN 10 AND 20 FOR UPDATE;
    

在上述例子中,如果没有间隙锁,事务1的第二次查询可能会读到事务2插入的新数据,导致不可预知的结果。通过使用FOR UPDATE和间隙锁,可以确保事务1在范围 [10, 20] 内的查询结果不会被其他事务插入新数据所影响。

需要注意的问题:

  1. 性能开销: 使用间隙锁可能会增加性能开销,因为它限制了其他事务在相同范围内插入数据。

  2. 并发控制: 间隙锁在一些情况下可能导致并发控制的降低,因此在设计时需要权衡并考虑具体的业务场景。

  3. 事务隔离级别: 间隙锁的行为可能会受到事务隔离级别的影响,需要谨慎选择适当的隔离级别。

在并发操作中,使用间隙锁能够确保对特定范围内的记录进行独占性操作,从而维护数据的一致性和完整性。

记录锁

记录锁(Row-level lock)是一种锁定数据库表中单个记录的机制。它在事务中的实际应用场景中发挥关键作用,可以保护数据的完整性。以下是记录锁的实际应用场景以及在事务中如何使用记录锁的详细讨论:

实际应用场景:

  1. 更新操作:

    • 当一个事务要对某个记录进行更新时,可以使用记录锁确保其他事务不能同时修改相同的记录,防止并发更新导致数据不一致。
    -- 事务1
    START TRANSACTION;
    SELECT * FROM your_table WHERE your_condition FOR UPDATE;
    -- 执行更新操作
    UPDATE your_table SET your_column = 'new_value' WHERE your_condition;
    COMMIT;-- 事务2
    START TRANSACTION;
    SELECT * FROM your_table WHERE your_condition FOR UPDATE; -- 会等待事务1释放锁
    -- 执行更新操作
    UPDATE your_table SET your_column = 'another_value' WHERE your_condition;
    COMMIT;
    
  2. 插入操作:

    • 当一个事务要在某个范围内插入新记录时,可以使用记录锁防止其他事务在相同范围内插入数据,避免幻读问题。
    -- 事务1
    START TRANSACTION;
    SELECT * FROM your_table WHERE your_column BETWEEN 10 AND 20 FOR UPDATE;
    -- 执行插入操作
    INSERT INTO your_table (your_column) VALUES (15);
    COMMIT;-- 事务2
    START TRANSACTION;
    SELECT * FROM your_table WHERE your_column BETWEEN 10 AND 20 FOR UPDATE; -- 会等待事务1释放锁
    -- 执行插入操作
    INSERT INTO your_table (your_column) VALUES (18);
    COMMIT;
    
  3. 删除操作:

    • 当一个事务要删除某个记录时,可以使用记录锁确保其他事务不能同时访问和修改相同的记录。
    -- 事务1
    START TRANSACTION;
    SELECT * FROM your_table WHERE your_condition FOR UPDATE;
    -- 执行删除操作
    DELETE FROM your_table WHERE your_condition;
    COMMIT;-- 事务2
    START TRANSACTION;
    SELECT * FROM your_table WHERE your_condition FOR UPDATE; -- 会等待事务1释放锁
    -- 执行其他操作
    COMMIT;
    

注意事项:

  1. 记录锁是通过使用FOR UPDATE语句实现的,它会锁定查询结果集中的行,防止其他事务在同一行上执行写操作。

  2. 记录锁的使用需要谨慎,过度使用可能导致性能问题,因此在设计时需要根据实际情况进行权衡。

  3. 事务隔离级别的选择会影响记录锁的行为,需要根据业务需求选择合适的隔离级别。

  4. 记录锁的释放通常发生在事务提交时,因此事务的持有时间应该尽量短,以减小锁的粒度和持有时间。

在事务中使用记录锁可以确保并发事务对数据库表中的记录进行独占性操作,从而维护数据的完整性。

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

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

相关文章

Python 读取创建word文档

本篇文章内容为使用python 读取word文档和创建word文档 读取doc文件 引入类库 示例如下: import win32com import win32com.client import os 读取doc文件 通过得到的doc文件路径调用系统word功能。 打开文件获取其中的文本信息,输出文本信息&#…

彩虹全新 SUP 模板,卡卡云模板修复版

彩虹全新 SUP 模板,卡卡云模板,首页美化,登陆页美化,修复了 PC 端购物车页面显示不正常的问题。下载地址:彩虹全新 SUP 模板.zip 模板截图

[ Python+OpenCV+Mediapipe ] 实现对象识别

一、写在前面 本文所用例子为个人学习的小结,如有不足之处请各位多多海涵,欢迎小伙伴一起学习进步,如果想法可在评论区指出,我会尽快回复您,不胜感激! 所公布代码或截图均为运行成功后展示。 二、本文内容…

python 条件语句if else

python 条件语句if else Python条件语句是通过一条或多条语句的执行结果(True或者False)来决定执行的代码块。 可以通过下图来简单了解条件语句的执行过程: Python程序语言指定任何非0和非空(null)值为true,0 或者…

【C语言基础】:操作符详解(一)

文章目录 操作符详解1. 操作符的分类2. 二进制和进制转换2.1 什么是二进制、八进制、十进制、十六进制2.1.1 二进制和进制转换2.1.2 二进制转十进制2.2.3 二进制转八进制2.2.4 二进制转十六进制 3. 源码、反码、补码4. 移位操作符4.1 左移操作符4.2 右移操作符 5. 位操作符&…

Java学习笔记------继承

继承 Java中提供了一个关键字extends,用这个关键字,我们可以让一个类和另一个类建立继承关系 如图,Student和Teacher类中除了study()和teacher()两个成员函数不同,其他重复了&…

【摸鱼日常】使用Docker部署2048小游戏

一、本次实践介绍 ​1. 本次实践简介 本次实践部署环境为个人测试环境,快速使用docker部署2048小游戏。 rootWellDone:/home/goodjob# uname -aLinux WellDone 6.5.0-14-generic #14~22.04.1-Ubuntu SMP PREEMPT_DYNAMIC Mon Nov 20 18:15:30 UTC 2 x86_64 x86_64…

【深度学习笔记】3_2线性回归的从零实现

注:本文为《动手学深度学习》开源内容,仅为个人学习记录,无抄袭搬运意图 3.2 线性回归的从零开始实现 在了解了线性回归的背景知识之后,现在我们可以动手实现它了。尽管强大的深度学习框架可以减少大量重复性工作,但若…

【C++精简版回顾】12.友元函数

1.友元函数 1.class class MM { public:MM(int age,string name):age(age),name(name){}friend void print(MM mm); private:int age;string name;void print() {cout << age << "岁的" << name << "喜欢你" << endl;} }; f…

Flutter(一):安装和环境配置、创建Flutter项目

安装和环境配置、创建Flutter项目 Flutter 下载方式1方式2 Flutter 环境配置配置国内镜像站点解压 Flutter将 flutter 添加到系统环境变量中运行 flutter doctor来验证安装 Android Studio下载插件创建项目安装 Android SDK 工具在模拟器上运行 Flutter 下载 方式1 全版本&…

Java基于微信小程序的校园二手物品交易系统,附源码

博主介绍&#xff1a;✌程序员徐师兄、7年大厂程序员经历。全网粉丝12w、csdn博客专家、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精彩专栏推荐订阅&#x1f447;…

计网:动手尝试SMTP交互【利用Telnet发送邮件, 带图片】

文章目录 准备工作发送仅有ascii码的邮件发送图片附件后记 准备工作 1.如图&#xff0c;勾选telnet客户端 2.邮箱开启第三方登录服务 开启服务后&#xff0c;会给一个授权码。授权码是QQ邮箱用于登录第三方客户端/服务的专用密码&#xff0c;适用于登录以下服务&#xff1a;…

React组件详解

React组件分为两大类 1.函数组件 2.类组件&#xff08;最常用&#xff09; 组件化 import ReactDom from "react-dom";// // 1.通过函数创建一个组件 // 2.函数名字必须大写开头 // 3.函数必须有返回值 function Func1() {return <h2>这是一个基础组件</h…

c语言经典测试题5

1.题1 t0; while(printf("*")) { t; if (t<3) break; }关于上述代码描述正确的是&#xff1f; A: 其中循环控制表达式与0等价 B: 其中循环控制表达式与0等价 C: 其中循环控制表达式是不合法的 D: 以上说法都不对 我们来分析一下&#xff1a;printf的返回值…

你真的了解@Async吗?

你真的了解Async吗&#xff1f; 使用场景&#xff1a; 开发中会碰到一些耗时较长或者不需要立即得到执行结果的逻辑&#xff0c;比如消息推送、商品同步等都可以使用异步方法&#xff0c;这时我们可以用到Async。但是直接使用 Async 会有风险&#xff0c;当我们没有指定线程池…

数学建模资料分享

1. 往年各赛题的优秀论文 可以用来参考一下论文是怎么写的。参考论文的结构&#xff0c;格式&#xff0c;思路等等。 链接&#xff1a;https://pan.baidu.com/s/1WG2t4-x9MjtaSgkq4ue5AQ?pwdnlzx 提取码&#xff1a;nlzx --来自百度网盘超级会员V4的分享 2.论文模板 链接&a…

typescript 实现Optional

我们先看下面的这段代码,一个学生接口,里面有成员id,name,age,gender等等成员, 有一个方法graduate,里面要接受一个Student类型的实参 interface Student {id: numbername: stringage: numbergender: string}function graduate(Student: Student) {//...}现在有一个问题,就是学…

LabVIEW燃料电池船舶电力推进监控系统

LabVIEW燃料电池船舶电力推进监控系统 随着全球经济一体化的推进&#xff0c;航运业的发展显得尤为重要&#xff0c;大约80%的世界贸易依靠海上运输实现。传统的船舶推进系统主要依赖于柴油机&#xff0c;这不仅耗能高&#xff0c;而且排放严重&#xff0c;对资源和环境的影响…

【笔记】【电子科大 离散数学】 2.命题

文章目录 数理逻辑定义 命题定义不是命题的例子 原子命题和复合命题定义约定 命题联结词否定联结词定义例子真值表 合取联结词定义例子真值表 析取联结词定义例子 蕴含联结词定义例子真值表 等价联结词定义例子真值表 命题符号化及其应用速查表格优先级复合命题符号化布尔检索演…

15.4K Star,超强在线编辑器

Hi&#xff0c;骚年&#xff0c;我是大 G&#xff0c;公众号「GitHub指北」会推荐 GitHub 上有趣有用的项目&#xff0c;一分钟 get 一个优秀的开源项目&#xff0c;挖掘开源的价值&#xff0c;欢迎关注。 今天推荐一款非常棒的开源实时协作编辑器&#xff0c;可用于多人同时编…