以前在面试准备期间写过相关的笔记,但时间仓促,未进行详述,对你们参考意义不大。面试
在MySQL
数据库面试/开发中,事务一直是重中之重,这些知识在以前面试的时候学过,到如今也记得不太清楚了,本文带领你们一块儿来回顾MySQL
的事务知识。数据库
事务一般并发执行,为了不事务并发形成的一系列问题,数据库设计四大事务隔离级别:并发
MySQL
默认的事务隔离级别是REPEATABLE-READ
,Oracle
默认的
事务隔离级别是READ-COMMITTED
。数据库设计
在MySQL
的众多引擎中,只有InnoDB
引擎支持事务与外键,而另外一经常使用的MyISAM
引擎虽不支持事务、外键,但访问速度快。测试
下图来自:《深刻浅出MySQL
数据库开发优化与管理维护》优化
咱们聊聊最难理解的REPEATABLE-READ
级别。spa
在READ-COMMITED
隔离级别下,会出现如下问题:设计
事务A | 事务B |
---|---|
开启事务 | 开启事务 |
查询帐户余额为3000元 | / |
/ | 更新帐户余额为3500元 |
/ | 提交事务 |
使用相同的条件去查询帐户余额,发现余额变为了3500元 | / |
提交事务 | / |
事务A
就很蒙圈,两次查询结果不同,我以哪一个为准呢?要是以3500
为准,那若是3500
也被人改了呢?事务A
不知道咋办了。3d
事务中屡次读取结果不一致,这种现象为不可重复读。code
新建一张基于InnoDB
引擎的表account
。
经过SELECT @@tx_isolation;
查询当前事务隔离级别。
去Navicat
开启用于测试的查询Session
:
事务A | 事务B | 事务C |
---|---|---|
START TRANSACTION; | / | / |
SELECT * FROM account WHERE id = 1; | / | / |
![]() |
/ | / |
/ | START TRANSACTION; | / |
/ | UPDATE account SET balance = 3500 WHERE id = 1; | / |
/ | COMMIT; | / |
/ | / | START TRANSACTION; |
SELECT * FROM account WHERE id = 1; | / | SELECT * FROM account WHERE id = 1; |
![]() |
/ | ![]() |
COMMIT; | / | COMMIT; |
经过本示例说明,普通的SELECT
:SELECT * FROM account WHERE id = 1
,没有加锁。由于若是加了锁,InnoDB
行级锁下会拒绝写操做。
出现问题了,虽然解决了不可重复读的问题,可是以上帝视角来看,以3000
进行运算,最终的结果仍是错误的。
这时候就须要锁了。
共享锁(读锁):读时加锁,容许其余事务共享锁,但不容许其余事务获取排它锁。
排它锁(写锁):写时加锁,不容许其余事务获取共享、排它锁。
共享锁、排它锁,也成为读、写锁,锁为何这么设计不用详细描述相信你们也能理解。
DML
:数据库操做语言(不要小看概念,有人面试就挂在DML
上了),也就是咱们经常使用的SELECT、INSERT、UPDATE、DELETE
等操做语句。
在InnoDB
引擎下,默认INSERT、UPDATE、DELETE
会对相关记录加行级排它锁,默认SELECT
不加锁。
可经过lock in share mode
加共享锁,for update
加排它锁。
共享锁示例:
事务A | 事务B |
---|---|
START TRANSACTION; | START TRANSACTION; |
SELECT * FROM account WHERE id = 1 LOCK IN SHARE MODE; | / |
![]() |
/ |
/ | SELECT * FROM account WHERE id = 1; |
/ | 不加锁查询:![]() |
/ | SELECT * FROM account WHERE id = 1 LOCK IN SHARE MODE; |
/ | 其余事务拥有共享锁,本事务获取共享锁成功:![]() |
/ | SELECT * FROM account WHERE id = 1 FOR UPDATE; |
/ | 其余事务拥有共享锁,本事务获取排它锁失败:![]() |
/ | UPDATE account SET balance = 3500 WHERE id = 1; |
/ | UPDATE语句获取排它锁,一样失败:![]() |
COMMIT; | / |
/ | COMMIT; |
排它锁示例:
事务A | 事务B |
---|---|
START TRANSACTION; | START TRANSACTION; |
SELECT * FROM account WHERE id = 1 FOR UPDATE; | / |
![]() |
/ |
/ | SELECT * FROM account WHERE id = 1; |
/ | 不加锁查询:![]() |
/ | SELECT * FROM account WHERE id = 1 LOCK IN SHARE MODE; |
/ | 其余事务拥有排它锁,本事务获取共享锁失败:![]() |
/ | SELECT * FROM account WHERE id = 1 FOR UPDATE; |
/ | 其余事务拥有排它锁,本事务获取排它锁失败:![]() |
/ | UPDATE account SET balance = 3500 WHERE id = 1; |
/ | UPDATE语句获取排它锁,一样失败:![]() |
COMMIT; | / |
/ | COMMIT; |
最后,提醒你们签约慎重,警戒互联网厂商毁约,避免被公司演一手。