前言:
好久没更新了,我又回来了。承接上一篇mysql innodb的文章。
隔离级别:
事务隔离级别要实际解决的问题
- 脏读
脏读指的是读到了其他事务未提交的数据,未提交意味着这些数据可能会回滚,也就是可能最终不会存到数据库中,也就是不存在的数据。读到了并一定最终存在的数据,这就是脏读。
-
更新丢失
一个事务的更新操作会被另外一个事务的更新操作所覆盖,从而导致数据的不一致 -
不可重复读
不可重复读指的是在同一事务内,不同的时刻读到的同一批数据可能是不一样的,可能会受到其他事务的影响,比如其他事务改了这批数据并提交了。通常针对数据更新(UPDATE)操作。
- 幻读
幻读是针对数据插入(INSERT)操作来说的。假设事务A对某些行的内容作了更改,但是还未提交,此时事务B插入了与事务A更改前的记录相同的记录行,并且在事务A提交之前先提交了,而这时,在事务A中查询,会发现好像刚刚的更改对于某些数据未起作用,但其实是事务B刚插入进来的,让用户感觉很魔幻,感觉出现了幻觉,这就叫幻读。
各个事务隔离级别对问题的解决
Read UnCommitted (读未提交)
事务中的修改,即使未提交,对于其他事务也是可见的。
原理:
对于读-写不进行加锁或者mvcc控制,所以导致可见。
作为最低级别的隔离级别
Read Committed (读已提交)
一个事务只能读取已经提交的事务所做的修改,即使一个事务所做的修改在提交之前对于其他事务都是不可见的。
原理:
每个 select 语句都有自己的一份快照,而不是一个事务一份,所以在不同的时刻,查询出来的数据可能是不一致的。利用了mvcc多版本并发控制
Read Repeatable (可重复读)
保证在同一个事务中多次读取同一数据的结果是一样的
原理:
一个事务一份读取快照,保证了在同一事务中读取的同一数据的结果都是一样的。利用了mvcc多版本并发控制
默认来说解决了幻读:对于聚簇索引采用行级锁,其他字段的查询/修改采用了next-key锁,防止一定范围内的别的事务进行的插入操作。
Serializable (事务串行化)
强制事务串行执行,这样多个事务互不干扰。读的时候加共享锁,也就是其他事务可以并发读,但是不能写。写的时候加排它锁,其他事务不能并发写也不能并发读。
原理:
将事务的执行变为顺序执行,与其他三个隔离级别相比,它就相当于单线程,后一个事务的执行必须等待前一个事务结束。