MySQL中的锁
共享锁
共享锁也成为读锁,针对同一份数据,多个事务的读操作可以同时进行而不会互相影响,相互不阻塞的。
- 通过下面命令加共享锁
SELECT...LOCK IN SHARE MODE
#或
SELECT...FOR SHARE;#(8.0新增语法)
排他锁
排他锁也叫写锁,当一个事务对一份数据执行写入,即加上排他锁后,其他事务对同一份数据进行读写操作会阻塞,直到前一个事务提交。
- 通过下面的命令加排他锁
SELECT ... FOR UPDATE;
DELETE、UPDATE、INSERT等操作也相当于加排他锁
共享锁和排他锁是否会发生阻塞如下图所示:
| 共享锁 | 排他锁 | |
|---|---|---|
| 共享锁 | 不阻塞 | 阻塞 |
| 排他锁 | 阻塞 | 阻塞 |
行级锁
行级锁是锁住行,粒度小,性能较高,但是行级锁只在存储引擎层实现,行级锁分为3种,记录锁、间隙锁和临键锁。
假如下面的一个表test_lock:
| id | a | b |
|---|---|---|
| 0 | 0 | 0 |
| 4 | 4 | 4 |
| 8 | 8 | 8 |
| 16 | 16 | 16 |
| 32 | 32 | 32 |
- id是主键索引
- a是普通索引
- b是普通列
记录锁 (Record Locks)
记录锁是仅仅锁住一条记录,锁的粒度最小。
什么时候会加记录锁?
当用唯一索引进行等职查询时,且查询的记录是存在的时候,会加记录锁。
| 会话1 | 会话2 | 会话3 |
|---|---|---|
| begin;select * from test_lock where id = 16 for update; | ||
| update test_lock set a = 100 where id = 16;(阻塞) | ||
| insert into test_lock value(9, 9, 9);(正常) |
- 会话1对
id=16记录加了行锁 - 会话2阻塞,无法对这条记录进行修改操作
- 会话3正常插入
间隙锁(Gap Locks)
"幻读"的问题,事务期间其他事务添加一条数据,再次读取突然多出一条记录。为了解决这样的问题,是不是可以对一段区间的数据加锁,加上锁以后,其他事务添加数据时必须阻塞。像这样的锁就叫做间隙锁,即锁定一个区间,左开右开。
什么情况会加 间隙锁 ?
- 在用唯一索引进行等值查询时,当查询记录不存在时,会加间隙锁。
select * from test_lock where id = 10 for update;
id=10位于8到16区间,由于10这条记录不存在,所以加的间隙锁,锁定(8, 16)的区间。
- 唯一锁引使用范围查询的时候,会加间歇锁。
select * from test_lock where id <10 and id> 8 for update;
id <10 and id> 8是一个范围查询,会锁定范围(8,16)。
- 普通索引等值查询时,如果记录存在,会额外添加一个间隙锁。
select * from test_lock where a = 8 for update;
由于a=8记录存在,会对范围(4,8]添加临键锁,这个后面会提到,同时额外向下遍历到第一个不符合条件的值才能停止,因此间隙锁的范围是(8,16)。
- 通索引等值查询时,如果记录不存在,会加一个间隙锁。
select * from test_lock where a = 10 for update;
此种情况锁定的范围为(8,16)
临键锁(Next-Key Locks)
如果想要同时集合上面的记录锁和间隙锁,也就是既想锁住某条记录,又想阻止其他事务在该记录前边的间隙插入新记录,所以InnoDB就提出了临键锁(Next-Key Locks),默认锁定的范围是左开右闭。InnoDB存储引擎默认的锁单位就是临键锁(Next-Key Locks),怎么理解呢?
也就是锁加锁都是按照临键锁加锁,但是会根据一定的规律退化为记录锁和间隙锁。具体规律如下:
唯一索引等值查询:
- 当查询的记录是存在的,临键锁会退化成「记录锁」。
- 当查询的记录是不存在的,临键锁 会退化成「间隙锁」。
非唯一索引等值查询:
- 当查询的记录存在时,除了会加 临键锁外,还额外加间隙锁,也就是会加两把锁。
- 当查询的记录不存在时,只会加 临键锁,然后会退化为间隙锁,也就是只会加一把锁。
InnoDB存储引擎中,如果一个表查询或者更新没有走索引,这时候还会创建行级锁吗? 答案是不会,这时候会升级为表级锁。
页级锁
页级锁是 MySQL 中锁定粒度介于行级锁和表级锁中间的一种锁。表级锁速度快,但冲突多,行级冲突少,但速度慢。因此,采取了折衷的页级锁,一次锁定相邻的一组记录。BDB (BerkeleyDB)存储引擎 支持页级锁。
特点
- 开销和加锁时间界于表锁和行锁之间
- 会出现死锁
- 锁定粒度界于表锁和行锁之间,并发度一般
表级锁
表锁会锁定整张表,它是MySQL中最基本的锁策略,并不依赖于存储引擎,表锁是开销最小的策略。因为表级锁一次会将整个表锁定,所以可以很好的避免死锁问题。但是表锁的并发度很差,那表锁都有哪几种呢?
表级别的S锁、X锁
LOCK TABLES t READ:InnoDB存储引擎会对表 t 加表级别的 S锁 。LOCK TABLES t WRITE:InnoDB存储引擎会对表 t 加表级别的 X锁 。
意向锁
意向锁也是一种表锁,表示某个事务正在锁定一行或者将要锁定一行,表明一个意图。它不与行级锁冲突。那它究竟有啥作用?
意向锁是在当事务加表锁时发挥作用。比如一个事务想要对表加排他锁,如果没有意向锁的话,那么该事务在加锁前需要判断当前表的每一行是否已经加了锁,如果表很大,遍历每行进行判断需要耗费大量的时间。如果使用意向锁的话,那么加表锁前,只需要判断当前表是否有意向锁即可,这样加快了对表锁的处理速度。
意向锁分为两种:
- 意向共享锁(intention shared lock, IS):事务有意向对表中的某些行加共享锁(S锁)
- 意向排他锁(intention exclusive lock, IX):事务有意向对表中的某些行加排他锁(X锁)
自增锁(AUTO-INC LOCK)
我们都知道在使用创建表的时候有自增主键AUTO_INCREMENT属性,那它是怎么实现自增的呢?
AUTO-INC锁是当向使用含有AUTO_INCREMENT列的表中插入数据时需要获取的一种特殊的表级锁,在执行插入语句时就在表级别加一个AUTO-INC锁,然后为每条待插入记录的AUTO_INCREMENT修饰的列分配递增的值,在该语句执行结束后,再把AUTO-INC锁释放掉。
元数据锁(MDL锁)
元数据锁可以用来保证读写的正确性。比如,如果一个查询正在遍历一个表中的数据而执行期间另一个线程对这个 表结构做变更 ,增加了一列,那么查询线程拿到的结果跟表结构对不上,肯定是不行的。
- 当对一个表做增删改查操作的时候,加 MDL读锁;
- 当要对表做结构变更操作的时候,加 MDL 写锁。
MDL读锁和读锁之间可以共享兼容,读锁和写锁之间不兼容,会互相阻塞。
全局锁
全局锁就是对 整个数据库实例 加锁。当你需要让整个库处于 只读状态 的时候,可以使用这个命令,之后 其他线程的以下语句会被阻塞:数据更新语句(数据的增删改)、数据定义语句(包括建表、修改表结 构等)和更新类事务的提交语句。
全局锁的典型使用场是做全库逻辑备份。
## 全局锁命令
flush tables with read lock
悲观锁
悲观锁是总以最坏的情况假设,比如操作一条数据,总认为也有其他线程要拿这条数据,那就给这条数据上排他锁,让其他事务或者线程阻塞,类似于Java中 synchronized 和 ReentrantLock 等独占锁的思想。
适用场景:
悲观锁适写操作多的场景,因为写的操作具有 排它性 。采用悲观锁的方式,可以在数据库层 面阻止其他事务对该数据的操作权限,防止读 - 写和写 - 写的冲突。
实现思路:
以秒杀商品为例,为了防止超卖,需要加锁。
#第1步: for update 方式查出商品库存
select quantity from items where id 1001 for update;
#第2步:如果库存大于0,则根据商品信息生产订单
insert into orders (item_id)values(1001);
#第3步:修改商品的库存,um表示购买数量
update items set quantity quantity-num where id 1001;
select····for update是MySQL中悲观锁。此时在items表中,id为1001的那条数据就被我们锁定了,其他的要执行select quantity from items where id=1001 for update;语句的事务必须等本次事务提交之后才能执行。这样我们可以保证每次事务能拿到最新的库存数量,从而不会超卖,但是这样的性能很差。
乐观锁
乐观锁认为一个事务发生并发的概率很小,就不加通过数据库加锁实现,因为加锁性能比较差,而是通过程序实现,那如何数据没有被其他事务修改了呢?会在更新数据的时候判断数据的版本或者时间戳是否发生变化。
实现思路:
版本号机制实现乐观锁
- 数据表中新增一个version字段
- 更新前读取出version字段
- 进行业务逻辑操作,更新数据,
UPDATE ... SET version=version+1 WHERE version=version,版本+1 - 如果有其他线程更新了数据,那么上面的修改不成功,返回值为0,表示已经被人更新了,我们可以根据0去做业务操作
时间戳机制实现乐观锁
时间戳和版本号机制一样,也是在更新提交的时候,将当前数据的时间戳和更新之前取得的时间戳进行 比较,如果两者一致则更新成功,否则就是版本冲突。
适用场景:
乐观锁适合读操作多的场景,相对来说写的操作比较少。它的优点在于 程序实现 ,不存在死锁问题。
相关文章:
MySQL中的锁
共享锁 共享锁也成为读锁,针对同一份数据,多个事务的读操作可以同时进行而不会互相影响,相互不阻塞的。 通过下面命令加共享锁 SELECT...LOCK IN SHARE MODE #或 SELECT...FOR SHARE;#(8.0新增语法)排他锁 排他锁也叫写锁,当一…...
WebView自定义进度条、加载动画,拿走直接用~
年前有个小需求,要对有些域名的H5进行加载流程优化,通过展示H5加载动画来安抚用户焦躁的心情,以提高用户体验。虽然不能理解加个动画咋就优化了用户体验,但需求还是得做的。想着这是个基础的小功能,独立性比较好&#…...
内存数据库Apache Derby、H2
概述 传统关系型数据库涉及大量的工作,如果想在Java应用程序里使用MySQL数据库,至少需要如下步骤: 安装(可选:配置用户名密码)建表(要么从命令行进入,要么安装一个可视化工具&…...
麻省理工出版 | 2023年最新深度学习综述手册
UCL Simon Prince的新书:《Understanding Deep Learning》 ,在2023年2月6日由MIT Press出版。他之前写过很受欢迎的《Computer Vision: Models, Learning, and Inference》。 关于这本最新的深度学习手册,作者这样介绍它: 正如书…...
vi命令详解
VIM - Vi IMproved 7.4 (2013 Aug 10, compiled Oct 13 2020 16:04:38) 用法: vim [参数] [文件 …] 编辑指定的文件 或: vim [参数] - 从标准输入(stdin)读取文本 或: vim [参数] -t tag 编辑 tag 定义处的文件 或: vim [参数] -q [errorfile] 编辑第一个出错处的文件 参数:…...
抖音的外卖行业入局,为中小外卖企业创业者的机会给了哪些机会?
一则关于抖音进入外卖市场的消息,让美团“非常受伤”。 2月8日,美团(03690.HK)盘中跌幅超9%。截至收盘,美团报收153.1港元,跌幅6.48%。美团大幅下跌的根源就是前一天关于抖音外卖进展的消息传闻。 2月7日,…...
供应PEG试剂AC-PEG-COOH,Acrylate-PEG-Acid,丙烯酸酯-PEG-羧基
英文名称:AC-PEG-COOH,Acrylate-PEG-Acid 中文名称:丙烯酸酯-聚乙二醇-羧基 丙烯酸酯-PEG-COOH是一种含有丙烯酸酯和羧酸的线性杂双功能PEG试剂。它是一种有用的带有PEG间隔基的交联剂。丙烯酸酯可与紫外光或自由基引发剂聚合。丙烯酸酯-PE…...
java二叉排序树
1.先看一个需求 给你一个数列 (7, 3, 10, 12, 5, 1, 9),要求能够高效的完成对数据的查询和添加 2.解决方案分析 使用数组 数组未排序, 优点:直接在数组尾添加,速度快。 缺点:查找速度慢. [示意图] 数组排序…...
聊一聊 gRPC 的四种通信模式
温馨提示:本文需要结合上一篇 gRPC 文章一起食用,否则可能看不懂。 前面一篇文章松哥和大家聊了 gRPC 的基本用法,今天我们再来稍微深入一点点,来看下 gRPC 中四种不同的通信模式。 gRPC 中四种不同的通信模式分别是:…...
科技云报道:开源真的香,风险知多少?
科技云报道原创。 过去几年,开源界一片火热,开源软件技术已全面进军操作系统、云原生、人工智能、大数据、半导体、物联网等行业领域。 数据显示,我国超九成企业在使用或正计划使用开源技术。 与此同时,全球各大开源组织相继兴…...
国产化适配迁移记录
国产化适配迁移记录 本项目基于RuoYi-Vue的框架进行迁移。目前已完成覆盖测试暂无其他问题。 国产化环境 名称版本达梦数据库DmJdbcDriver18 8.1.2.144通用mapper – tk.mybatismapper-spring-boot-starter 4.2.5<!-- 达梦数据库--><dependency><groupId>…...
又一国产开源项目走向世界,百度RPC框架Apache bRPC正式成为ASF顶级项目
2023 年 1 月 26 日,Apache 软件基金会 (ASF) 官方正式宣布Apache bRPC 正式毕业,成为 Apache的顶级项目。 我听到这个消息是挺开心的,毕竟是又一款由国人主导的apche顶级项目,再次证明国内在开源界正在发挥越来越重要的作用。 …...
多数据库学习之GBase8s查询数据库表元信息常用SQL
多数据库学习之GBase8s查询数据库表元信息常用SQL简介常用SQL创建用户创建数据库及模式获取表元数据其他参考链接简介 背景介绍 GBase 8t是基于IBM informix源代码、编译和测试体系自主研发的交易型数据库产品。 南大通用安全数据库管理系统(简称 GBase 8sÿ…...
Jetpack之Lifecycle应用与源码分析
Build lifecycle-aware components that can adjust behavior based on the current lifecycle state of an activity or fragment. 上面是源于官网的定义,简单翻译就是说Lifecycle的作用就是基于当前的Activity或者Fragment的生命周期当前状态构建可感知生命周期的…...
Python序列类型之集合
💐💐💐欢迎来到小十一的博客!!! 🎯博客主页:🎯程序员小十一的博客 🚀博客专栏:🚀Python入门基础语法 🌷欢迎关注ÿ…...
java 自定义json解析注解 复杂json解析
java 自定义json解析注解 复杂json解析 工具类 目录java 自定义json解析注解 复杂json解析 工具类1.背景2、需求-各式各样的json一、一星难度json【json对象中不分层】二、二星难度json【json对象中出现层级】三、三星难度json【json对象中存在数组】四、四星难度json【json对象…...
Vue3配置路由(vue-router)
文章目录前言一、配置路由(vue-router)1、安装路由2、新建页面3、创建路由配置文件4.特殊报错!前言 紧接上篇文章,vue3的配置与vue2是有所差别的,本文就讲述了如何配置,如果本文对你有所帮助请三连支持博主…...
【代码随想录二刷】Day9-字符串-C++
代码随想录二刷Day9 今日任务 28.找出字符串中第一个匹配项的下标 459.重复的子字符串 字符串总结 双指针总结 语言:C KMP 链接:https://programmercarl.com/0459.重复的子字符串.html#kmp 用处:当出现字符串不匹配时,可以利…...
google colab上如何下载bert相关模型
首先要知道模型的地址 tensorflow版本的模型: https://storage.googleapis.com/bert_models/2018_10_18/cased_L-12_H-768_A-12.zip https://storage.googleapis.com/bert_models/2018_11_03/chinese_L-12_H-768_A-12.zip pytorch版本的模型 ‘bert-base-cased’: …...
Vue2.0页面缓存机制联合页面标签的交互(keep-alive + router)
预期效果:(借助iview-ui的在线体验页面示意一下) 项目中只有一部分页面需要缓存,且存在多级路由的页面。每打开一个菜单,就会新增一个 Tab标签,只要 Tab标签不关闭,对应的页面就会被缓存&#x…...
突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合
强化学习(Reinforcement Learning, RL)是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程,然后使用强化学习的Actor-Critic机制(中文译作“知行互动”机制),逐步迭代求解…...
React第五十七节 Router中RouterProvider使用详解及注意事项
前言 在 React Router v6.4 中,RouterProvider 是一个核心组件,用于提供基于数据路由(data routers)的新型路由方案。 它替代了传统的 <BrowserRouter>,支持更强大的数据加载和操作功能(如 loader 和…...
python/java环境配置
环境变量放一起 python: 1.首先下载Python Python下载地址:Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个,然后自定义,全选 可以把前4个选上 3.环境配置 1)搜高级系统设置 2…...
el-switch文字内置
el-switch文字内置 效果 vue <div style"color:#ffffff;font-size:14px;float:left;margin-bottom:5px;margin-right:5px;">自动加载</div> <el-switch v-model"value" active-color"#3E99FB" inactive-color"#DCDFE6"…...
spring:实例工厂方法获取bean
spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂ÿ…...
解决本地部署 SmolVLM2 大语言模型运行 flash-attn 报错
出现的问题 安装 flash-attn 会一直卡在 build 那一步或者运行报错 解决办法 是因为你安装的 flash-attn 版本没有对应上,所以报错,到 https://github.com/Dao-AILab/flash-attention/releases 下载对应版本,cu、torch、cp 的版本一定要对…...
select、poll、epoll 与 Reactor 模式
在高并发网络编程领域,高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表,以及基于它们实现的 Reactor 模式,为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。 一、I…...
项目部署到Linux上时遇到的错误(Redis,MySQL,无法正确连接,地址占用问题)
Redis无法正确连接 在运行jar包时出现了这样的错误 查询得知问题核心在于Redis连接失败,具体原因是客户端发送了密码认证请求,但Redis服务器未设置密码 1.为Redis设置密码(匹配客户端配置) 步骤: 1).修…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...
【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...
