MySQL事务之隔离性

概念

当数据库上有多个事务同时执行的时候,就可能出现脏读(dirty read)不可重复读(non-repeatable read)幻读(phantom read)的问题,为了解决这些问题,就有了"隔离级别"的概念。

隔离得越严实,效率就会越低。要在二者之间寻找一个平衡点。

隔离级别

读未提交

一个事务还没提交时,它做的变更就能被别的事务看到。

读已提交

一个事务提交之后,它做的变更才会被其他事务看到。

可重复读

一个事务执行过程中看到的数据,总是跟这个事务在启动时看到的数据是一致的。当然在可重复读隔离级别下,未提交变更对其他事务也是不可见的。

串行化

顾名思义是对于同一行记录,会加写锁会加读锁。当出现读写锁冲突的时候,后访问的事务必须等前一个事务执行完成,才能继续执行。

命令

查看

mysql> select @@GLOBAL.tx_isolation,@@session.tx_isolation;
+-----------------------+------------------------+
| @@GLOBAL.tx_isolation | @@session.tx_isolation |
+-----------------------+------------------------+
| REPEATABLE-READ       | REPEATABLE-READ        |
+-----------------------+------------------------+
1 row in set, 2 warnings (0.10 sec)

设置

mysql> set session transaction isolation level read uncommitted;
Query OK, 0 rows affected (0.00 sec)

mysql> set session transaction isolation level read committed;
Query OK, 0 rows affected (0.00 sec)

mysql> set session transaction isolation level repeatable read;
Query OK, 0 rows affected (0.00 sec)

mysql> set session transaction isolation level serializable;
Query OK, 0 rows affected (0.00 sec)

case

mysql> create table T(c int) engine=InnoDB;
insert into T(c) values(1);
事务 A事务 B
启动事务;查询得到值 1启动事务
查询得到值 1
将 1 改成 2
查询得到值 V1
提交事务
查询得到值 V2
提交事务
查询得到值 V3
  • 若隔离级别是读未提交, 则V1的值就是 2 。这时候事务B虽然还没有提交,但是结果已经被事务A看到了。因此,V2V3也都是 2

  • 若隔离级别是读提交,则V11V2的值是 2事务B的更新在提交后才能被事务A看到。所以, V3的值也是 2

  • 若隔离级别是可重复读,则V1V21V32 。之所以V2还是 1 ,遵循的就是这个要求:事务在执行期间看到的数据前后必须是一致的。

  • 若隔离级别是串行化,则在事务B执行将 1 改成 2 的时候,会被锁住。直到事务A提交后,事务B才可以继续执行。所以从事务A的角度看, V1V2值是 1V3的值是 2

数据库里面会创建一个视图,访问的时候以视图的逻辑结果为准。在可重复读隔离级别下,这个视图是在事务启动时创建的,整个事务存在期间都用这个视图。在读提交隔离级别下,这个视图是在每个 SQL 语句开始执行的时候创建的。读未提交隔离级别下直接返回记录上的最新值,没有视图概念。而串行化隔离级别下直接用加锁的方式来避免并行访问。

Oracle 数据库的默认隔离级别其实就是读提交,因此对于一些从 Oracle 迁移到 MySQL 的应用,为保证数据库隔离级别的一致,一定要记得将 MySQL 的隔离级别设置为读提交

并发问题

  • 脏读:对于两个事务T1, T2, T1读取了已经被T2更新但还没有被提交的字段. 之后, 若T2 回滚, T1读取的内容就是临时且无效的
  • 不可重复读:对于两个事务T1, T2, T1读取了一个字段, 然后T2 更新了该字段提交. 之后, T1再次读取同一个字段, 值就不同了(在一个事物中不管读多少次,读取的数据应该都一样)
  • 幻读:对于两个事务T1, T2, T1从一个表中读取了一个字段, 然后T2在该表中插入了一些新的行. 之后, 如果T1再次读取同一个表, 就会多出几行
脏读不可重复读幻读
读未提交
读已提交×
可重复读××
串行化×××
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值