数据库事务简单介绍
数据库事务简单介绍
事务和锁
一、什么是事务
事务是逻辑上的一组数据库操作,要么都执行,要么都不执行
举例:假如张三给李四转账300,那么这笔转账涉及两个关键操作:1、张三的银行余额要减少300。2、李四的余额里增加300。如果两个操作突然出现错误,比如银行系统出错导致张三余额减少了但是李四余额并没有增加。这样的系统是存在问题的,事务是保证关键操作要么都成功,要么都失败。
事务的特性
- 1、原子性:事务是最小的执行单位,不允许分割。事务的原子性确保动作要么全部完成,要么完全不起作用;例如转账的这两个关键操作(将张三的余额减少200元,将李四的余额增加200元)要么全部完成,要么全部失败。
- 2、一致性: 确保从一个正确的状态转换到另外一个正确的状态,这就是一致性。例如转账业务中,将张三的余额减少200元,中间发生断电情况,李四的余额没有增加200元,这个就是不正确的状态,违反一致性。又比如表更新事务,一部分数据更新了,但一部分数据没有更新,这也是违反一致性的。
- 3、隔离性:并发访问数据库时,一个用户的事务不被其他事务所干扰,各并发事务之间数据库是独立的。
- 4、持久性:一个事务被提交之后,对数据库中数据的改变是持久的,即使数据库发生故障也不应该对其有任何影响。
二、事务之间的相互影响
事务之间的相互影响分为几种,分别为:脏读,不可重复读,幻读,丢失更新
脏读
举例:老板给程序员发工资,程序员的工资是1万/月。但是发工资时老板不小心按错了数字,按成1.5万/月,该钱已经打到程序员的账户,但是事务还没有提交,就在这时,程序员去查看自己这个月的工资,发现比往常多了5千元,以为涨工资了非常高兴。但是老板及时发现了不对,马上回滚差点就提交了的事务,将数字改成1万再提交。
当事务T1正在访问字段A并且对其进行了修改,而这种修改还没有提交到数据库中(操作在提交数据之前)。这时另外一个事务T2也访问和使用字段A,但由于事务T1修改字段A后还没有提交 COMMIT,而那么事务T2读到的字段A是脏数据。
丢失更新
第一类丢失更新:撤销一个事务的时候,把其他事务已提交的更新数据覆盖了。这是完全没有事务隔离级别造成的,如果事务1被提交,另一个事务被撤销,那么会连同事务1所做的更新也被撤销。
第二类丢失更新:当两个或多个事务查询相同的记录,然后各自基于查询的结果更新记录时会造成第二类丢失更新问题。每个事务不知道其它事务的存在,最后一个事务对记录所做的更改将覆盖其它事务之前对该记录所做的更改。
举例:事务T1读取A=20,事务T2也读取A=20,事务T1修改A=A-1,事务T2也修改 A=A+1,最终结果A=21,事务T1的修改被丢失。
不可重复读
举例:程序员拿着信用卡去享受生活(卡里有1万),当他买单时(程序员事务开启),收费系统事先检测到他的卡里有1万,就在这时!!程序员的妻子要把钱全部转出充当家用,并提交。当收费系统准备扣款时,再检测卡里的金额,发现已经没钱了(第二次检测金额当然要等待妻子转出金额事务提交完)。程序员就会很郁闷,明明卡里是有钱的
不可重复读取是指同一个事务在整个事务过程中对同一笔数据进行读取,每次读取结果都不同。如果事务1在事务2的更新操作之前读取一次数据,在事务2的更新操作之后再读取同一笔数据一次,两次结果是不同的。
不可重复读出现的原因就是事务并发修改记录,要避免这种情况,最简单的方法就是对要修改的记录加锁,这回导致锁竞争加剧,影响性能。
幻读
举例:程序员某一天去消费,花了2千元,然后他的妻子去查看他今天的消费记录(全表扫描FTS,妻子事务开启),看到确实是花了2千元,就在这个时候,程序员花了1万买了一部电脑,即新增INSERT了一条消费记录,并提交。当妻子打印程序员的消费记录清单时(妻子事务提交),发现花了1.2万元,似乎出现了幻觉。
在同一个事务中,同一个查询多次返回的结果不一致。事务A新增了一条记录,事务B在事务A提交前后各执行了一次查询操作,发现后一次比前一次多了一条记录, 就好像发生了幻觉一样。
幻读是由于并发事务增加记录导致的,这个不能像不可重复读通过记录加锁解决,因为对于新增的记录根本无法加锁。需要将事务串行化,才能避免幻读。
三、数据库隔离级别及原理
| 隔离级别 | 脏读 | 不可重复读 | 幻读 |
|---|---|---|---|
| 读未提交 | Y | Y | Y |
| 读已提交 | - | Y | Y |
| 可重复读 | - | - | Y |
| 看序列化 | - | - | - |
- 标题: 数据库事务简单介绍
- 作者: Xxd
- 创建于 : 2023-08-09 21:52:50
- 更新于 : 2024-02-07 03:57:05
- 链接: https://blog.xxdoge.site/2023/08/09/数据库事务/
- 版权声明: 本文章采用 CC BY-NC-SA 4.0 进行许可。