MySQL事务隔离级别:读未提交、读已提交、可重复读和串行

什么是事务隔离级别?

在进行多个事务的并发执行时,如果不对它们进行隔离,则可能会产生一些问题。例如:脏读、不可重复读和幻读。而事务隔离级别就是用来解决这些问题的。

MySQL中定义了四种事务隔离级别,不同的隔离级别会导致不同的并发执行结果。在实际应用中,需要根据业务的特点选择合适的隔离级别。MySQL的四种事务隔离级别依次为:读未提交(Read Uncommitted)、读已提交(Read Committed)、可重复读(Repeatable Read)和串行化(Serializable)。

读未提交(Read Uncommitted)

读未提交是最低的隔离级别,允许一个事务读取并使用另一个事务尚未提交的修改。因此,在该级别下可能会发生脏读问题。

脏读是指在并发执行的两个事务中,一个事务读到了另一个事务尚未提交的数据。在读未提交的情况下,如果一个事务对数据进行了修改,但是还没有提交,则另一个事务读取该数据时可能会得到错误的结果。这样就会造成脏读问题。

示例1:

事务A更新表t1中的数据并未提交:

begin;
update t1 set name='aaa' where id=1;

事务B读取表t1中的数据:

select * from t1 where id=1;

此时,事务B可能会读到 name列值为’aaa’ 的行,而在事务A提交之前,name列实际上是没有被更新的。

因此,读未提交级别并不安全,不建议使用。

读已提交(Read Committed)

在读已提交级别下,一个事务只能读取到已经提交的其他事务所修改过的数据。因此,该级别解决了脏读问题。

但是,在该级别下仍然存在不可重复读和幻读问题。

不可重复读是指在同一个事务中,由于其他事务的干扰,导致同一查询语句返回的结果不同。

示例2:

事务A从表t1中读取数据:

begin;
select * from t1 where id=1;

在A事务还未提交之前,事务B修改了表t1中的数据:

begin;
update t1 set name='bbb' where id=1;
commit;

此时,当事务A再次执行相同的查询语句时,得到的结果已经不同了。

幻读是指在同一个事务中,由于其他事务的干扰,导致同一查询条件下返回的行集合不同。

示例3:

事务A从表t1中读取数据:

begin;
select * from t1 where name like '%a%';

在A事务还未提交之前,事务B向表t1中插入了一些数据:

begin;
insert into t1 (name) values ('abc'), ('def');
commit;

此时,当事务A再次执行相同的查询语句时,得到的结果已经不同了。

因此,针对不可重复读和幻读问题,需要使用更高的隔离级别。

可重复读(Repeatable Read)

在可重复读级别下,一个事务在执行期间多次读取同一行数据,将得到相同的结果。因此,在该级别下解决了不可重复读问题。

但是,仍然存在幻读问题。

解决幻读问题需要引入行锁,MySQL中提供了next-key lock来实现。

next-key lock是指对于一个索引的范围进行加锁,以避免出现幻读问题。

串行化(Serializable)

串行化是最高的隔离级别,它强制事务串行执行,避免了脏读、不可重复读和幻读等问题。在该级别下,MySQL会对所有读取的数据行都加共享锁或排他锁,直到事务结束。

由于串行化对性能的影响比较大,所以一般情况下不建议使用。只有在确实需要完全隔离、对并发度要求不高的业务场景下才使用。

总结

MySQL提供了四种事务隔离级别,读未提交是最低的级别,因为它存在脏读问题。读已提交解决了脏读问题,但是仍然存在不可重复读和幻读问题。可重复读解决了不可重复读问题,但是仍然存在幻读问题。串行化解决了所有并发问题,但会对性能产生极大的影响。

在实际应用中,需要根据业务的特点选择合适的隔离级别,以保证数据的正确性和并发性。

Guess you like

Origin blog.csdn.net/weixin_43025343/article/details/131864919