面试官:谈一下对MySQL事务隔离的理解

什么是事务隔离?

隔离是什么?看到"隔离"这个字眼,估计大部分人的第一反应就是"WC,阳了",那么"隔离你"就是为了保证其他人的安全。

MySQL中的事务隔离也是这样的,主要是为了确保在并发事务执行时,各个事务之间能够相互独立、互不干扰地运行,从而保证数据的一致性。

事务的隔离级别

MySQL事务隔离为了满足不同场景,提供了4个事务隔离级别(严格来讲是InnoDB存储引擎支持的),分别是:

  • 读未提交(Read Uncommitted)
  • 读已提交(Read Committed)
  • 可重复读(Repeatable Read)
  • 串行化(Serializable)

读未提交

读未提交,很好理解:事务在各自的周期内,可以读取到其他事务未提交的数据。

以下图为例,事务2的两次查询结果分别为18和19,即便事务1没有提交。

这种现象被称为"脏读"。

读已提交

读已提交,很好理解:事务在各自的周期内,每次对同一个sql进行查询都是已经提交的数据。

以下图为例,事务2的两次查询结果分别为18和19,因为在此期间事务1对数据进行了更改。

但是,这种现象被称为"不可重复读",因为违反了事务的隔离性原则(在一个事务内,数据应该是一致的)

个人认为,在大部分情况下,可以读到已经提交过的数据,就不算是一个问题(MySQL默认的隔离级别也是这个)。

可重复读

可重复读,这样理解:既然有"不可重复读"的现象,那我满足你,让你可以重复读。

即:每个事务在自己的事务周期内,多次对同一个sql进行查询都是相同的数据。

以下图为例,事务2的两次查询结果都一样,即便在此期间事务1对数据进行了更改。

但是,如果在"可重复读 "级别下进行了当前读 ,那么还是会出现像"读已提交"中出现的"不可重复读"的现象,不过这里称之为"幻读"(叫法真多)。

这里说下什么是当前读。

顾名思义,当前读就是读取行数据当前最新的状态,这个过程中会对所有读、写的数据加上相应的读锁、写锁、范围锁,目的是避免并发冲突。在MySQL中使用以下sql语句都属于当前读:

sql 复制代码
UPDATE ...
sql 复制代码
DELETE ...
sql 复制代码
INSERT ...
sql 复制代码
SELECT ... LOCK IN SHARE MODE
sql 复制代码
SELECT ... FOR UPDATE

可能会有人问:UPDATEDELETEINSERT为什么是读操作?

《MySQL是如何保证数据不丢失的?》中有介绍过,在执行DML操作时,会先根据条件去查询相应的行数据,所以也存在读操作。

与当前读对应的还有快照读。快照读是一种不加锁的查询,目的是为了提高并发性能。(具体实现原理,将在后续文章中为您揭晓)

普通的SELECT语句,都属于快照读,只不过不同的隔离级别快照的时机不同。

串行化

可串行化,很好理解:并发事务的执行完全是串行,依次执行,不存在冲突导致的一致性问题。但是可想而知,这种隔离级别的并发效率肯定极为低下。

总结

MySQL提供了多种事务隔离级别,包括读未提交、读已提交、可重复读和串行化等。这些隔离级别在并发性能和数据一致性之间提供了不同的权衡,可以根据具体的应用场景和需求来选择适合的隔离级别。

相关推荐
艾伦~耶格尔5 分钟前
【数据结构进阶】
java·开发语言·数据结构·学习·面试
smileNicky28 分钟前
SpringBoot系列之从繁琐配置到一键启动之旅
java·spring boot·后端
David爱编程1 小时前
为什么必须学并发编程?一文带你看懂从单线程到多线程的演进史
java·后端
愿天堂没有C++1 小时前
剑指offer第2版——面试题4:二维数组中的查找
c++·面试
long3161 小时前
java 策略模式 demo
java·开发语言·后端·spring·设计模式
AI 嗯啦2 小时前
SQL详细语法教程(三)mysql的函数知识
android·开发语言·数据库·python·sql·mysql
rannn_1113 小时前
【Javaweb学习|黑马笔记|Day1】初识,入门网页,HTML-CSS|常见的标签和样式|标题排版和样式、正文排版和样式
css·后端·学习·html·javaweb
柏油3 小时前
Spring @Cacheable 解读
redis·后端·spring
柏油3 小时前
Spring @TransactionalEventListener 解读
spring boot·后端·spring
杰克尼4 小时前
mysql-条件查询案例
数据库·mysql