数据库事务(Transaction)与锁(Locking)详解图析_事务与锁-程序员宅基地

技术标签: 数据库  

一、事务


事务(Transaction)是由一系列对系统中数据进⾏访问与更新的操作所组成的⼀个程序执行逻辑单元。
在这里插入图片描述
:中止(abort):表示事务未成功结束,撤消事务的所有操作。
数据库应用程序通常通过事务而不是单个操作访问数据库。例如,大型数据库和百万并发用户:银行、双十一、订票系统等。

结合程序语言的角度通过实例理解一下事务:
插入(INSERT)、选择(SELECT)、更新(UPDATE)、删除(DELETE)
开始(BEGIN)、提交(COMMIT)、中止(ABORT)/ 回滚(ROLLBACK)等;

BEGIN TRANSACTION
SELECT balance FROM summary WHERE name = `张三';
UPDATE summary SET balance = balance-500 WHERE name=`张三';
SELECT balance FROM summary WHERE name = `李四';
UPDATE summary SET balance = balance+500 WHERE name = `李四';
COMMIT

典型的银行转账张三转给李四张三-500,李四+500

内部进程级别:操作对象为数据库数据(表行列内存单元)。
读(read)、写(write)开始(begin)、提交(commit)
中止(abort):表示事务未成功结束,撤消事务的所有操作

步骤 Transactions
1 read(张三)
2 write(张三) (张三:= 张三- 500)
3 read(李四)
4 write(李四) (李四:= 李四+ 500)
5 commit

带着问题学习事务

  1. 事务的语法
  2. 事务的特性
  3. 事务的并发问题
  4. 事务的隔离级别
  5. 不同隔离级别的锁的情况了解
  6. 隐式提交(了解)

结合下面的事务图了解:
在这里插入图片描述

1.1 事务的语法

  1. start transaction; begin;
  2. commit; 提交 使得当前的修改确认
  3. rollback; 回滚 使得当前的修改被放弃

1.2 事务的ACID特性

  1. 原⼦性(Atomicity)
    事务的原⼦性是指事务必须是⼀个原子的操作序列单元。事务中包含的各项操作在⼀次执⾏过程中,只允许出现两种状态之一。
    (1)全部执行成功
    (2)全部执行失败
    事务开始后所有操作,要么全部做完,要么全部不做,不可能停滞在中间环节。事务执⾏过程中出错,会回滚到事务开始前的状态,所有的操作就像没有发⽣一样。也就是说事务是⼀个不可分割的整体,就像化学中学过的原子,是物质构成的基本单位。
  2. ⼀致性(Consistency)
    事务的一致性是指事务的执⾏不能破坏数据库数据的完整性和一致性,一个事务在执⾏之前和执行之后,数据库都必须处以⼀致性状态。
    比如:如果从A账户转账到B账户,不可能因为A账户扣了钱,⽽B账户没有加钱。
  3. 隔离性(Isolation)
    事务的隔离性是指在并发环境中,并发的事务是互相隔离的。也就是说,不同的事务并发操作相同的数据时,每个事务都有各自完整的数据空间。⼀个事务内部的操作及使用的数据对其它并发事务是隔离的,并发执行的各个事务是不能互相干扰的。隔离性分4个级别
  4. 持久性(Duration)
    事务的持久性是指事务⼀旦提交后,数据库中的数据必须被永久的保存下来。即使服务器系统崩溃或服务器宕机等故障。只要数据库重新启动,那么一定能够将其恢复到事务成功结束后的状态。

1.3 事务的并发问题

  • 脏读:读取到了没有提交的数据, 事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据。
    在这里插入图片描述

  • 不可重复读:同⼀条命令返回不同的结果集(更新).事务 A 多次读取同一数据,事务 B 在事务A 多次读取的过程中,对数据做了更新并提交,导致事务A多次读取同一数据时,结果不一致。
    在这里插入图片描述

  • 幻读:重复查询的过程中,数据就发⽣了量的变化(insert,delete)。
    在这里插入图片描述

1.4 事务隔离级别

事务隔离级别 脏读 不可重复读 幻读
读未提交(READ_UNCOMMITTED) 允许 允许 允许
读已提交(READ_COMMITTED) 禁止 允许 允许
可重复读(REPEATABLE_READ) 禁止 禁止 可能会
顺序读(SERIALIZABLE) 禁止 禁止 禁止

4种事务隔离级别从上往下,级别越高,并发性越差,安全性就越来越高。 ⼀般数据默认级别是读以提交或可重复读

  • 读未提交(READ_UNCOMMITTED)
    读未提交,该隔离级别允许脏读取,其隔离级别是最低的。换句话说,如果一个事务正在处理理某一数据,并对其进⾏了更新,但同时尚未完成事务,因此还没有提交事务;而以此同时,允许另一个事务也能够访问该数据。
    脏读示例:
    在事务A和事务B同时执行时可能会出现如下场景:
时间 事务A(存储) 事务B(取款)
T1 开始事务
T2 开始事务
T3 查询余额(1000元)
T4 取出1000元(余额0元)
T5 查询余额(余额0元)
T6 撤销事务(余额恢复1000元)
T7 存入500元
T8 提交事务

余额应该为1500元才对。请看T5时间点,事务A此时查询的余额为0,这个数据就是脏数据,他是事务B造成的,很明显是事务没有进行隔离造成的。


  • 读已提交(READ_COMMITTED)
    读已提交是不同的事务执行的时候只能获取到已经提交的数据。 这样就不会出现上面的脏读的情况了。但是在同一个事务中执行同一个读取,结果不一致不可重复读示例
    可是解决了脏读问题,但是还是解决不了可重复读问题。
时间 事务A(存储) 事务B(取款)
T1 开始事务
T2 开始事务
T3 查询余额(1000元)
T4 查询余额(余额1000元)
T5 取出1000元(余额0元)
T6 提交事务
T7 查询余额(余额0元)
T8 提交事务

事务A其实除了查询两次以外,其它什么事情都没做,结果钱就从1000变成0了,这就是不不可重复读的问题。


  • 可重复读(REPEATABLE_READ)
    可重复读就是保证在事务处理过程中,多次读取同一个数据时,该数据的值和事务开始时刻是一致的。因此该事务级别限制了不可重复读和脏读,但是有可能出现幻读的数据。
    幻读就是指同样的事务操作,在前后两个时间段内执行对同一个数据项的读取,可能出现不一致的结果。诡异的更新事件。
时间 事务A(存储) 事务B(取款)
T1 开始事务
T2 查询当前所有数据 开始事务
T3 插入一条数据
T4 查询当前所有数据 提交事务
T5 进行范围修改
T6 查询当前所有数据
T7 提交事务

可以看出在T3中事务B插入了一条数据,重复查询的过程中,数据就发⽣了量的变化(insert,delete)。

  • 顺序读(SERIALIZABLE)
    顺序读是最严格的事务隔离级别。它要求所有的事务排队顺序执⾏,即事务只能一个接一个地处理,不能并发。

二、 锁(Locking)

先了解锁的概念

2.1 锁的概念与用法

锁是一种用于并发控制的技术,可保证事务的隔离性。锁在数据库中一般作用在对象上,如文件、表、记录、页等。

锁的用法分成两类:

  • 共享锁:多个事务可以同时获取它。
  • 互斥锁:只有一个事务可以获得它,导致其他试图获取它的事务等待。持有锁的事务完成后,它释放锁,允许一个等待的事务获取锁。

2.2 不同的隔离级别的锁的情况

  1. 读未提交(RU): 有行级的锁,没有间隙锁。它与RC的区别是能够查询到未提交的数据。
  2. 读已提交(RC):有行级的锁,没有间隙锁,读不到没有提交的数据。
  3. 可重复读(RR):有行级的锁,也有间隙锁,每次读取的数据都是一样的,并且没有幻读的情况。
  4. 序列化(S):有行级锁,也有间隙锁,读表的时候,就已经上锁了

2.3 行锁

行锁就是针对数据库中表的行记录的锁,这很好理解,比如事务 A 更新了一行,而这时候,事务 B 也要更新一行,则必须等事务 A 的操作完成后才能更新。
注:MyISAM 不支持行锁,InnoDB 是支持行锁的


2.4 两阶段锁(2PL:Two-phase Locking)

举一个实例,假设有book表,有bookid和name字段,在下面的操作中,事务 B 的 update 语句执行时,会是什么现象呢
在这里插入图片描述
这个问题的结论取决于事务 A 执行完前两条语句后,持有哪些锁,以及在什么时候释放。
实际上,事务 A 持有两个记录的行锁,都是在 commit 的时候才释放的,所以事务 B 的 update 就会被阻塞,直到事务 A 执行 commit 之后,事务 B 才能被继续执行。也就是说,在 InnoDB 事务中,行锁是在需要的时候才加上的,但并不是不需要了就立刻释放,需要等事务结束时才释放,这就是两阶段锁协议,分为加锁阶段和解锁阶段,所有的 lock 操作都在 unlock 操作之后


2.5 两阶段锁的优化

假设你负责实现一个医院缴费管理,患者A要在医院缴费,需要涉及以下操作:

语句1:扣除患者A 账户余额

语句2:增加医院总账户余额

语句3:记录一条交易日志

也就是说,完成这次交易,需要 update 两条记录,并 insert 一条记录。当然为了保证交易的原子性,我们需要这三个操作放在一个事务中。与此同时,还有患者B也需要在医院缴费,那么你会怎样安排这三个语句在事务中的顺序呢?

不管哪个患者都需要的步骤就是语句2,增加医院总账户余额,这两个事务都需要进行这个操作,根据两阶段协议,不论怎么安排语句,所有的操作需要的行锁都是在事务提交的时候才释放的,要想使行锁在事务中不会停留太长时间,最大程度的减少了事务之间的锁等待,节约资源,就应该把语句2直接放在最后如下图:
在这里插入图片描述

2.6 死锁

两个或多个事务的相互阻塞
如下图所示,事务 A 在等待事务 B 释放 id = 2 的行锁,而事务 B 在等待 事务 A 释放 id = 1 的行锁,事务 A 和事务 B 在互相等待对方的资源释放,就是进入了死锁状态。
在这里插入图片描述
死锁不是数据库自身的问题,我们无法通过优化数据库配置来解决或者避免死锁,只能通过修改应用程序来解决。简单来说,我们应该在程序中按照相同的顺序修改数据,避免产生相互等待资源的情况发生。
不过,我们在实际应用中可能无法完全按照相同顺序修改数据。如果出现了不可避免的死锁情况,另一种解决方法就是捕获系统返回的死锁异常并在程序中加入重试机制。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_45670060/article/details/119977481

智能推荐

史上最全Redis面试题及答案-程序员宅基地

文章浏览阅读1w次,点赞7次,收藏93次。在网上看到有关Redis的50道面试题目,但是没有给出答案,之前我也在寻找这份试题的答案,特地把答案分享出来。有需要的可以看看咯花了大量时间整理了这套Redis面试题首发50题,绝无仅有,从入门到精通从基础,高级知识点,再到集群,运维,方案…弄明白了这些题可以说可以成为面霸了面试官都得折服,Redis学得怎么样,都来检验下吧..._redis面试题

KMP算法之next数组计算详细解析_kmp算法next计算方法-程序员宅基地

文章浏览阅读8.6k次,点赞10次,收藏56次。KMP算法相比BF算法的改进:每当一趟匹配过程中出现字符比较不等时,无需回溯i指针(即无需将i指针完全退回至i-j+1),而是利用已经得到的“部分匹配”的结果将模式向右“滑动”尽可能远的一段距离后,继续进行比较。需要解决的问题:当主串中的第i个字符与模式中第j个字符比较不相等时,主串中第i个字符(i指针不回溯)应与模式中哪个字符再比较?----假设从主串中第i个字符与模式中的第k个字符再进行比较它是则呢样来消除回溯的呢?就是因为它提取并运用了加速匹配的信息!  这种信息就是对于每模式串 t 的每个_kmp算法next计算方法

yshon对讲机如何调频率_对讲机频率如何设置?-程序员宅基地

文章浏览阅读524次。展开全部TK2118-3118调频率方法1. 按住MONI键+DIAL键开e68a84e8a2ad3231313335323631343130323136353331333363396461机至显示SELF;2. 按一下LOW显CH1,转动频道旋钮"ENC"选择所需信道;3. 按一下PTT键显------2,按一下LOW键显示接收频率,按住 "1"键转动频道旋钮"ENC"调整数,松开 "1"键转动..._yshon对讲机设置

Linux安装Git并配置完成拉取项目代码和提交(1)-程序员宅基地

文章浏览阅读747次,点赞21次,收藏19次。本人从事网路安全工作12年,曾在2个大厂工作过,安全服务、售后服务、售前、攻防比赛、安全讲师、销售经理等职位都做过,对这个行业了解比较全面。最近遍览了各种网络安全类的文章,内容参差不齐,其中不伐有大佬倾力教学,也有各种不良机构浑水摸鱼,在收到几条私信,发现大家对一套完整的系统的网络安全从学习路线到学习资料,甚至是工具有着不小的需求。最后,我将这部分内容融会贯通成了一套282G的网络安全资料包,所有类目条理清晰,知识点层层递进,需要的小伙伴可以点击下方小卡片领取哦!

数模3—Matlab线性规划、非线性规划、多目标规划(超全解法合集)-程序员宅基地

文章浏览阅读1.9w次,点赞49次,收藏461次。生产两种机床,利润分别为XXX, A机器和B机器加工,两种机器工作时间…若不存在不等式约束,用"[]”代替A和b: [x, fval ] = linprog (f,[],[], Aeq, beq, lb, ub)若不存在等式约束,用"[]”代替Aeq和beq: [x, fval ] = linprog Cf,A,b,[],[] ,lb,ub)A机器和B机器加工,有顺序要求,有不同损耗费用,不同的工作时间…每种资产的平均收益率…结论,4个人攻碉楼,2个人追替身的情况下,百姓的士气最大,最大值为220。_线性规划

stm32f103运行python_microPython 有没有在 STM32F103上运行的固件-程序员宅基地

文章浏览阅读1.7k次。匿名用户1级2017-02-20 回答进入第一个个连接进入之后我们就可以选择红色方框Buid Firmware。之后我们就可以下载相应的固件了,但是这个固件是比较早的就是了,是2015年的固件,所以我们接下来要做的就是来自己编译一个固件。首先我们需要将我们的micropython的github开源项目的内容下载下来。选择下图的图标就是可以下载下来。或者在window或者linux底下直接使用Git..._micropython stm32f103

随便推点

python symbols函数_Python的高级特征你知多少?来对比看看-程序员宅基地

文章浏览阅读81次。Python 多好用不用多说,大家看看自己用的语言就知道了。但是 Python 隐藏的高级功能你都 get 了吗?本文中,作者列举了 Python 中五种略高级的特征以及它们的使用方法,快来一探究竟吧!Python 是一种美丽的语言,它简单易用却非常强大。但你真的会用 Python 的所有功能吗?任何编程语言的高级特征通常都是通过大量的使用经验才发现的。比如你在编写一个复杂的项目,并在 stack..._python symbols函数与lambda函数

人类高质量 Java 学习路线【一条龙版】_鱼皮学习路线-程序员宅基地

文章浏览阅读2.3w次,点赞135次,收藏780次。Java 学习路线一条龙版 by 鱼皮。原创不易,请勿抄袭,违者必究!大家好,我是鱼皮。现在网上的编程资料实在太多了,而且人人肯定都说自己的最好,那就导致大家又不知道怎么选了。大部分的博主推荐资源,也就是把播放量高的视频说一遍,水一期视频,没有一条很清晰的学习路线。所以今天我的这个 Java 学习路线就做做减法,给大家来个一条龙服务,Java 要学的知识点、对应的最佳学习资源和预计要花费的时间,都安排的明明白白的,不用选了,有计划了,也别再迷茫和纠结了,就无脑跟着学就行了。我还在文档中整理了链接._鱼皮学习路线

oracle的教材,oracle基础入门教材-程序员宅基地

文章浏览阅读83次。建表create table (column_name1 column_type [not null] [check (expression)][default value] [primary key][references (column_name)],column_name2 column_type [not null] [check (expression)][default value] ..._oracle最新教材

MariaDB数据库安装与使用_使用提供的数据库软件包安装数据库mariadb。使用root用户,登录数据库,将反馈信息-程序员宅基地

文章浏览阅读410次。MariaDB数据库管理系统安装yum install mariadb mariadb-server开启systemctl start mariadb开机启动systemctl enable mariadb初始化mariaDB服务操作涉及下面5个步骤。1 设置root管理员在数据库中的密码值(默认空值,直接回车)2 设置root管理员在数据库中的专有密码。(设置密码)3 随后删除匿名账户,并使用r..._使用提供的数据库软件包安装数据库mariadb。使用root用户,登录数据库,将反馈信息

BGP防环机制_allow-as-loop-程序员宅基地

文章浏览阅读2.5k次。EBGP之间使用AS号来防环,可通过命令allow-as-loop来解除。IBGP水平分割:从IBGP邻居学到的BGP路由不发送给IBGP邻居路由反射器中的:originator-id和cluster-id 起源id和集群id路由聚合时会自动产生指向null的路由(无论手动和自动都会)IBGP学到的路由默认不能引入到IGP中( EBGP可以引入),可通过命令配置。用实验来模拟一下最后一个,IBGP学到的路由默认不能引入IGP中,可以用命令修改,import-bgp ibgp。AR2和._allow-as-loop

【个人作品】记之-串口日志记录工具-程序员宅基地

文章浏览阅读1.3k次,点赞2次,收藏2次。一款用于记录嵌入式设备的串口调试信息设备,目前可以同时支持两路串口输入数据,且串口参数可配置。