数据库事务(transaction)与锁(locking)详解图析(代码片段)

小炭在努力 小炭在努力     2022-12-27     410

关键词:

一、事务


事务(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
1read(张三)
2write(张三) (张三:= 张三- 500)
3read(李四)
4write(李四) (李四:= 李四+ 500)
5commit

带着问题学习事务

  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 在互相等待对方的资源释放,就是进入了死锁状态。

死锁不是数据库自身的问题,我们无法通过优化数据库配置来解决或者避免死锁,只能通过修改应用程序来解决。简单来说,我们应该在程序中按照相同的顺序修改数据,避免产生相互等待资源的情况发生。
不过,我们在实际应用中可能无法完全按照相同顺序修改数据。如果出现了不可避免的死锁情况,另一种解决方法就是捕获系统返回的死锁异常并在程序中加入重试机制。

数据库事务与锁

开启事务就自动加锁。事务与锁是不同的。事务具有ACID(原子性、一致性、隔离性和持久性),锁是用于解决隔离性的一种机制。事务的隔离级别通过锁的机制来实现。另外锁有不同的粒度,同时事务也是有不同的隔离级别的。一... 查看详情

事务4.3-事务与锁(锁)

数据库和操作系统一样,是一个多用户使用的共享资源。当多个用户并发地存取数据时,在数据库中就会产生多个事务同时存取同一数据的情况。若对并发操作不加控制就可能会读取和存储不正确的数据,破坏数据库的一致性。... 查看详情

事务4-事务与锁

...化Serializable)。在具体的程序设计中,开启事务其实是要数据库支持才行的 查看详情

mysql事务与锁

数据库一般都会并发执行多个事务,多个事务可能会并发的对相同的一批数据进行增删改查操作,可能就会导致我们说的脏写、脏读、不可重复读、幻读这些问题。这些问题的本质都是数据库的多事务并发问题,为了解决多事务... 查看详情

数据库事务隔离级别与锁

...本特征     所谓事务是用户定义的一个数据库操作序列,这些操作要么全做要么全不做,是一个不可分割的工作单位。例如,在关系数据库中,一个事务可以是一条SQL语句、一组SQL语句或整个程序。事务ACID特性。ACID... 查看详情

mysqldump--single-transaction和--lock-tables参数详解

...导出,拼接成insert语句的形式进行备份。 关于--single-transaction和--lock-tables--single-transaction选项和--lock-tables选项是互斥的,因为LOCKTABLES会使任何挂起的事务隐含提交 相关探究实验1.开启mysqlg 查看详情

mysql事务与锁表的问题?

后台语言用php,数据库为mysql,用InnoDB引擎,我先开启事务,然后用locktables锁住所有会受影响的表。但我发现如果先开启事务,再锁表。出现错误时事务不起作用,不能回滚,但程序执行过程中表是锁定的。如果把锁表放在事务... 查看详情

线程安全与锁(代码片段)

线程安全线程安全是多线程编程时的计算机程序代码中的一个概念。在拥有共享数据的多条线程并行执行的程序中,线程安全的代码会通过同步机制保证各个线程都可以正常且正确的执行,不会出现数据污染等意外情况。多个线... 查看详情

spring@transactional——事务回滚

...下来的整个事务中,客户代码都应该使用该connection连接数据库,执行所有数据库命令[不使用该connection连接数据库 查看详情

关于fmdb事务(transaction)的解读

...要么都不执行。所以,应该把它们看成一个事务。事务是数据库维护数据一致性的单位,在每个事务结束时,都能保持数据一致性。针对上面的描述可以看 查看详情

事务(transaction)

什么是事务(Transaction)指访问并可能更新数据库中各种数据项的一个程序执行单元(unit)--也就是由多个sql语句组成,必须作为一个整体执行这些sql语句作为一个整体一起向系统提交,要么都执行、要么都不执行 为什么需要事务... 查看详情

获取在同一事务中使用@Transactional 插入的数据

】获取在同一事务中使用@Transactional插入的数据【英文标题】:Getthedatawhichisinsertedusing@Transactionalinthesametransaction【发布时间】:2016-07-0801:23:30【问题描述】:我在一种方法中插入数据(有@Transactional(propagation=Propagation.Required))但... 查看详情

spring@transactional——事务回滚

...下来的整个事务中,客户代码都应该使用该connection连接数据库,执行所有数据库命令[不使用该connection连接数据库 查看详情

spring事务transaction源码深度解析

目录一、基于xml形式开启Transaction1.创建数据库user2.创建一个maven项目 3.通过xml形式配置事务1)创建Spring命名空间2)开启事务配置3)创建UserService类4.测试事务1) 抛出RuntimeException 2)注释掉RuntimeException二、事务开启入口TxNamespaceHandler... 查看详情

mysql事务transaction(代码片段)

...除和该人员相关的信息,如信箱,文章等等,这样,这些数据库操作语句就构成一个事务!在MySQL中只有使用了Innodb数据库引擎的数据库或表才支持事务。事务处理可以用来维护数据库的完整性,保证成批的SQL语句要么全部执行... 查看详情

mybatis配置多个数据源事务(transaction)处理

  当mybatis配置文件中只有一个数据源的时候,按照正常的事务注解形式@Transaction是没有问题的,但是当配置文件中有多个数据源的时候发现事务不起作用了,怎么解决这个问题呢?看下面的案例: 查看详情

spring事务transaction源码深度解析(代码片段)

目录一、基于xml形式开启Transaction1.创建数据库user2.创建一个maven项目 3.通过xml形式配置事务1)创建Spring命名空间2)开启事务配置3)创建UserService类4.测试事务1) 抛出RuntimeException 2)注释掉RuntimeException二、事务开启入口TxNamespaceHandler... 查看详情

事务transaction那点事儿

...:原子性(Atomicity)。我靠,一点都不神秘嘛。特别是在数据库领域,事务是一个 查看详情