5分钟了解,Mysql事务事务隔离级别

01引言

上一节主要介绍了声明式事务,其中的一个属性就是事务的隔离级别isolation。今天我们以Mysql为例,深入介绍一下事务的隔离隔离级别。

02 隔离级别概述

事务的隔离级别,是为了解决并发问题而设定的。不同的隔离级别,并发条件下会发生不同的问题。主要问题有:

  • 脏读:读取到其他事务未提交的数据
  • 不可重复读:同一事务内多次读取同一条数据,因其他事务的修改或删除导致结果不一致。
  • 幻读:同一事务内多次按相同条件查询,行记录不一致

2.1 读未提交

READ UNCOMMITTED,读未提交,可以读到别人没有提交的事务数据。最低等级的隔离级别,属于无锁机制,自然性能也是最好的。但是容易产生脏读、不可重复读、幻读这些问题。这里级别问题较多,一般不会使用。

简单案例:

A将ID为1的记录的余额字段由原来的200改成了100,这个时候还没有提交事务。

这时,B又查询ID为1的记录,他查到的余额字段的结果就是100

2.2 读已提交

READ COMMITTED,读已提交,只能读到事务提交的数据。比READ UNCOMMITTED的隔离高一个等级,属于行级锁。可以避免脏读。但是无法避免不可重复读、幻读这些问题。

简单案例:

同读已提交的案例,A事务没有提交之前,B查询的结果就是200。只有A提交了事务,B才能读到余额200。

2.3 可重复读

REPEATABLE READ,可重复读,同一个事务中读取到的结果一致。比READ COMMITTED的隔离高一个等级,属于行级锁+间隙锁。可以避免脏读、不可重复读,但是无法解决幻读的问题。这个是Mysql默认的隔离级别。

SELECT @@transaction_isolation;

简单案例:

A先开启了事务,查询当天的数据库的记录数,假如是100条,这是时候还没有提交事务。

B新增了一条记录,并提交了事务。A这是再同一个事务中再次查询记录数,还是100条,看不到B新增的记录。

2.4 串行化

SERIALIZABLE,串行化。最严格的隔离级别。只要有事务发生,统统等待,一个一个顺序执行,完全避免并发问题。性能当然也是最差的。

03 案例分析

3.1 脏读

Dirty Read,脏读。在读未提交的隔离级别下发生,读取到其他事务未提交的数据,事务未提交的数据被称为脏数据。

-- 事务A(写)
START TRANSACTION;
UPDATE account SET balance = 1500 WHERE id = 1; -- 未提交

-- 事务B(读)
SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
START TRANSACTION;
SELECT balance FROM account WHERE id = 1; -- 读到1500(脏数据)

后果: 事务B基于无效数据决策(如允许转账),若事务A回滚导致数据不一致。

**解决方案:**只能升级数据库的隔离级别,升级至READ COMMITTED

3.2 不可重复读

Non-Repeatable Read, 不可重复读。在读未提交、读已提交的隔离级别下可能发生。

-- 事务A
SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
START TRANSACTION;
SELECT balance FROM account WHERE id = 1; -- 第一次:1000

-- 事务B提交更新
UPDATE account SET balance = 1500 WHERE id = 1;
COMMIT;

-- 事务A再次读取
SELECT balance FROM account WHERE id = 1; -- 结果变为1500!
COMMIT;

后果: 事务内相同查询结果不一致(如对账失败)。
解决方案: 升级隔离级别,使用REPEATABLE READ

3.3 幻读

Phantom Read,幻读。在读未提交、读已提交、可重复读的隔离级别下可能发生。

-- 事务A
START TRANSACTION; -- 默认REPEATABLE READ
SELECT * FROM account WHERE balance >= 1000; -- 返回id=1,2

-- 事务B插入新记录并提交
INSERT INTO account VALUES (3, 3000);
COMMIT;

-- 事务A再次查询
SELECT * FROM account WHERE balance >= 1000; 
-- 仍只返回id=1,2,3 → 出现"幻读"
COMMIT;

后果: 行记录的这种影响其实并不是太影响结果,大部分情况下都是可接受的。一般情况下是不需要的解决的。

3.4 不可重复读和幻读的区别

不可重复读和幻读都是在同一个事务下,多次获取的结果不同。但是又有根本的区别

  • 不可重复读:针对单条数据的修改冲突。

  • 幻读:针对结果集范围的新增或删除冲突,表现为行数增减或新行出现

04 小结

数据库的隔离级别影响着数据查询的结果,在满足业务需求的情况下,选择合适的隔离级别,可以提高数据的读写性能。但是我们一般选默认就够了,因为它已经足够好了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

智_永无止境

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值