面试官:谈一下对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提供了多种事务隔离级别,包括读未提交、读已提交、可重复读和串行化等。这些隔离级别在并发性能和数据一致性之间提供了不同的权衡,可以根据具体的应用场景和需求来选择适合的隔离级别。

相关推荐
字节跳动数据库9 分钟前
文章分享——相似函数处理方法
人工智能·后端·程序员
云技纵横10 分钟前
@Transactional 失效的 7 种场景:第 5 种最难排查
后端
用户67570498850227 分钟前
你知道 Go 结构体和结构体指针调用的区别吗?一文带你彻底搞懂!
后端·go
程序员cxuan43 分钟前
读懂 Claude Code 架构分析系列,第一篇,开始!
人工智能·后端·架构
用户6757049885021 小时前
面试官问“装饰器模式”,这样回答薪资多要 3000!
后端
tntxia1 小时前
Geo Scene域名修改引起的一些问题
后端
用户298698530141 小时前
Java 实现 Word 文档加密与权限解除
java·后端
vanuan1 小时前
给你的A2A-Agent加把锁-认证鉴权实战指南
后端
自由路飞1 小时前
RAG 混合检索深挖:BM25 和向量分数为什么不能直接相加?
面试
Yeats_Liao1 小时前
14:Servlet中的页面跳转-Java Web
java·后端·架构