【MySQL —— 事务】四大特性、隔离级别


文章目录

  • ​1. 什么是事务
  • 2. 为什么用事务
  • 3. 事务的使用步骤
  • 4. 事务的四大特性
    • 4.1 原子性(Atomicity)
    • 4.2 持久性(Durability)
    • 4.3 一致性(Consistency)
    • ⭐4.4 隔离性(重点)(Isolation)
  • 5. 并发执行事务的三大问题
    • 5.1 脏读问题
    • 5.2 不可重复读问题
    • 5.3 幻读问题
  • 6. MySQL 事务的隔离级别
    • 6.1 read uncommitted
    • 6.2 read committed
    • 6.3 repeatable read
    • 6.4 serializable

​1. 什么是事务

事务诞生的目的就是为了把若干个独立的操作打包成一个整体,这样的一组操作,要么全都执行,要么都不执行。事务是一个不可分割的数据库操作序列,也是数据库并发控制的基本单位。

2. 为什么用事务

以最典型的银行转账为例,A 给 B 转账,那么它的执行流程的是:A 账户 - 500 B 账户 + 500。万一在这两个操作之间突然出现错误比如银行系统崩溃,导致 A 余额减少而 B 的余额没有增加,这样就不对了。事务就是保证这两个关键操作要么都成功,要么都要失败。

3. 事务的使用步骤

  • 开启事务(start transaction)
  • 执行多条 SQL
  • 提交/回滚事务(commit 全部成功/rollback 全部失败)

代码示例:

-- 开启事务
start transaction;
-- A 账户减少500
update account set money = money - 500 where name = 'A';
-- B 账户加上500
update account set money = money + 500 where name = 'B';
-- 提交事务
commit;

4. 事务的四大特性

事务有四大特性,原子性、持久性、一致性和隔离性。

4.1 原子性(Atomicity)

一个事务中的所有操作,要么全部执行成功,要么全部执行失败。但是执行失败不是不执行,而是通过逆操作回滚数据,消除前面的 SQL
带来的影响。当第二个 SQL 执行失败时,数据库就会还原之前操作。

比如 A 账户 -500元的逆操作就是 A 账户 +500元。

4.2 持久性(Durability)

事务执行完成之后,它所做的所有修改都是永久的。因为事务一旦提交,数据就写入硬盘,持久化的存储起来了。

4.3 一致性(Consistency)

一个事务在执行前后数据必须保持一种合理合法的状态,事务总是从一个一致状态到另一个一致状态。

比如,两个账户进行转账,两个账户总额为100,转为账后总额也还是为100,不能使总额从100变为30,这样就是不合理的,不符合一致性了。

⭐4.4 隔离性(重点)(Isolation)

多个事务并发访问时,事务之间是相互隔离的,一个事务不应该被其他事务干扰,多个并发事务之间要相互隔离。隔离性解决的就是当并发执行多个事务,尤其是这个事务在尝试修改或者读取同一份数据,就会出现的一些问题。

并发执行事务可能出现的问题比如:当一个人正在写文章时,另一个人就去读,然后他就走开了,但是他读的不是写的最终结果,所以就会产生问题,这种就称为“脏读”问题
第二种情况就是,还是一个人在写文章,但是在他写的过程中给写这个操作进行了加锁,别人要等到他提交文章后才可以进行读操作。这样就避免了第一种情况出现的“脏读”问题。但是当别人进行读操作时,这个作者又开始进行写操作,最后读完时,就会出现读到了一些时没有修改之前的内容,又读到了修改之后的内容。这就是“不可重复读”问题
第三种情况是为了避免前两种情况,规定了修改的时候不能读,读的时候也不能修改。这样就解决了“脏读”问题和“不可重复读”问题。但是又会出现在你读的时候,他去写另一篇文章的情况,因为事务虽然在提交隔离性的时候要进行一系列的加锁,但它不是对所有表都进行加锁。这样当你读的时候就会发现多了一篇别的文章,这就是“幻读”问题

5. 并发执行事务的三大问题

5.1 脏读问题

脏读问题:事务 A 在对某个数据进行修改,修改的同时,事务 B 读取了数据,此时事务 B 读取的就是一个“脏数据”,它不是一个最终的结果。

5.2 不可重复读问题

不可重复读问题:在一个事务中,包含了多次的读操作,这多次读操作读出来的结果不一致。

5.3 幻读问题

幻读问题:一个事务执行过程中进行多次查询,多次查询的结果集不一样,这个操作也是一种特殊的不可重复读问题。

6. MySQL 事务的隔离级别

6.1 read uncommitted

允许读取未提交,该隔离级别的事务可以看到其他事务中未提交的数据。该隔离级别并发程度最高,隔离级别最低。此级别因为可以读取其他事务中未提交的数据,而未提交的数据可能会发生回滚,所以此级别读取到数据就叫做脏数据,也就是会发生脏读,还有不可重复读和幻读问题。

6.2 read committed

读取已提交,该隔离级别的事务只能读取到已提交事务的数据,因此解决了脏读问题,但是由于在事务执行中可以读取到其他事务提交的结果,所以在不同时间相同的 SQL 查询下会有不同的结果,这种就是不可重复读现象,还有幻读问题。在这个隔离级别中,并发程度降低了,但是隔离级别提高了,解决了脏读问题。

6.3 repeatable read

可重复读,是 MySQL 的默认事务隔离级别,它能确保同一事务多次查询的结果一致,但新的问题就是,别的事务正在读取时,另一个事务成功的插入某条数据,但是查询时又查不到这条数据,自己插入时又插不进去,也就是幻读问题。此种隔离级别,并发程度又降低了,隔离性又提高了,解决了不可重复读问题。

6.4 serializable

串行化,它会强制事务排序,使之不会发生冲突,解决了幻读问题。它是 MySQL 的最高隔离级别,但是并发程度最低,效率低,执行速度最慢,使用场景不多。

以上事务隔离级别可以通过修改 my.ini 这个配置文件来设置当前的数据库隔离级别,根据实际需求选择隔离级别。

活动地址:CSDN21天学习挑战赛

你可能感兴趣的:(数据库,mysql,数据库,java)