MySQL数据库InnnoDB引擎事务说明

前言

本篇文章主要讲诉数据库中事务的四大特性(ACID)以及事务的隔离级别划分。
 

数据库事务及其特性

事务是指满足ACID特性的一组操作,可以通过 Commit 提交一个事务,也可以使用 Rollback 进行回滚。对于MySQL的InnoDB引擎,其和MyIsAm引擎的主要区别就是InnoDB支持事务(题外话:InnoDB是MySQL5.5以后的默认执行引擎)。下面我们来介绍下事务的四大特性:
  • 原子性(Atomicity)
  原子性是指事务包含的所有操作要么全部成功,要么全部失败回滚,所以事务的操作如果成功就必须要完全应用到数据库,如果操作失败则不能对数据库有任何影响。为了实现原子性,需要通过日志将所有对数据的更新操作都写入日志,如果一个事务中的一部分操作已经成功,但以后的操作,由于断电/系统崩溃/其它的软硬件错误而无法继续,则通过回溯日志,将已经执行成功的操作撤销,从而达到“全部操作失败”的目的。
  • 一致性(Consistency)
  一致性是指事务必须使数据库从一个正确状态变换到另一个正确的状态,也就是说一个事务执行之前和执行之后都必须处于正确的状态。举例:拿转账来说,假设用户A有1000块钱,用户B有2000块钱,那么不管A和B之间如何转账,转几次账,事务结束后两个用户的钱相加起来应该还得是3000,这就是事务的一致性。事务不一致的情况:(1)用户A给用户B转500块钱,此时用户A账户减去500元,剩下500元,但是此时突然系统崩溃了,用户B没来得及加500,此时的数据库就出现了不一致的状态。(2)事务1需要将500元转入帐号A:先读取帐号A的值,然后在这个值上加上500。但是,在这两个操作之间,另一个事务2修改了帐号A的值,为它增加了500元。那么最后的结果应该是A增加了1000元。但事实上,事务1最终完成后,帐号A只增加了500元,因为事务2的修改结果被事务1覆盖掉了。这使得三个账户转账后,数据的不一致。
  • 隔离性(Isolation)
  隔离性是当多个用户并发访问数据库时,比如操作同一张表时,数据库为每一个用户开启的事务,不能被其他事务的操作所干扰,多个并发事务之间要相互隔离。
  即要达到这么一种效果:对于任意两个并发的事务T1和T2,在事务T1看来,T2要么在T1开始之前就已经结束,要么在T1结束之后才开始,这样每个事务都感觉不到有其他事务在并发地执行。
  关于事务的隔离性数据库提供了多种隔离级别,稍后会介绍到。
  • 持久性(Durability)
  持久性是指一个事务一旦被提交了,那么对数据库中的数据的改变就是永久性的,即便是在数据库系统遇到故障的情况下也不会丢失提交事务的操作。
  例如我们在使用JDBC操作数据库时,在提交事务方法后,提示用户事务操作完成,当我们程序执行完成直到看到提示后,就可以认定事务以及正确提交,即使这时候数据库出现了问题,也必须要将我们的事务完全执行完成,否则就会造成我们看到提示事务处理完毕,但是数据库因为故障而没有执行事务的重大错误。
 
ACID 之间的关系
用一句话描述ACID的关系就是:原子性,隔离性和持久性都是为了保证数据库数据的一致性
 
 

隔离级别

介绍完事务的四大特性(简称ACID)后,现在重点来说明下事务的隔离性,当多个线程都开启事务操作数据库中的数据时,数据库系统要能进行隔离操作,以保证各个线程获取数据的准确性,在介绍数据库提供的各种隔离级别之前,我们先看看如果不考虑事务的隔离性,会发生的几种问题:
1. 脏读
脏读是指并发过程中,一个事务处理过程里读取了另一个未提交的事务中的数据。
当一个事务正在多次修改某个数据,而在这个事务中这多次的修改都还未提交,这时一个并发的事务来访问该数据,就会造成两个事务得到的数据不一致。例如:用户A向用户B转账100元,对应SQL命令如下:
update account set money=money+100 where name=’B’; (此时A通知B) update account set money=money - 100 where name=’A’;
当只执行第一条SQL时,A通知B查看账户,B发现确实钱已到账(此时即发生了脏读),而之后无论第二条SQL是否执行,只要该事务不提交,则所有操作都将回滚,那么当B以后再次查看账户时就会发现钱其实并没有转。
2. 不可重复读
 不可重复读是指在对于数据库中的某条数据,一个事务范围内多次查询却返回了不同的数据值,这是由于在查询间隔,被另一个事务修改并提交了。例如事务T1在读取某一数据,而事务T2立马修改了这个数据并且提交事务给数据库,事务T1再次读取该数据就得到了不同的结果,发生了不可重复读。
不可重复读和脏读的区别是:
脏读是某一事务读取了另一个事务未提交的脏数据,而不可重复读则是两次读取之间存在另一个事务提交的数据。
 在某些情况下,不可重复读并不是问题,比如我们多次查询某个数据当然以最后查询得到的结果为主。但在另一些情况下就有可能发生问题,例如对于同一个数据A和B依次查询就可能不同,A和B就可能打起来了……
3. 幻读
 幻读是事务非独立执行时发生的一种现象。例如事务T1查询整张表中有多少条记录,这时事务T2又对这个表中插入了一行数据。而操作事务T1的用户如果再查看整张表有多少行数据,会发现多出一行数据,其实这行是事务T2添加的,就好像产生幻觉一样,这就是发生了幻读。
  幻读和不可重复读区别是:
For locking reads (SELECT with FOR UPDATE or LOCK IN SHARE MODE), UPDATE, and DELETE statements, locking depends on whether the statement uses a unique index with a unique search condition, or a range-type search condition.
(引用自MySQL官方手册:https://dev.mysql.com/doc/refman/5.7/en/innodb-transaction-isolation-levels.html)
  • 不可重复读的重点是修改(update)或者删除(delete),操作的是某一行数据,需要锁行。同样的条件, 你读取过的数据, 再次读取出来发现值不一样了。
  • 幻读的重点在于新增(insert),操作的是整张表,需要锁表。同样的条件, 第1次和第2次读出来的记录数不一样 

四种隔离级别

在数据库操作中,为了有效保证并发读取数据的正确性,提出的事务隔离级别,MySQL数据库为我们提供的四种隔离级别。
  1. 未提交读(Read Uncommitted):允许脏读,也就是可能读取到其他会话中未提交事务修改的数据
  2. 提交读(Read Committed):只能读取到已经提交的数据。Oracle等多数数据库默认都是该级别 (不重复读)
  3. 可重复读(Repeated Read):可重复读。在同一个事务内的查询都是事务开始时刻一致的,InnoDB默认级别。在SQL标准中,该隔离级别消除了不可重复读,但是还存在幻象读
  4. 串行读(Serializable):完全串行化的读,每次读都需要获得表级共享锁,读写相互都会阻塞
以上四种隔离级别最高的是Serializable级别,最低的是Read uncommitted级别,当然级别越高,执行效率就越低。像Serializable这样的级别,就是以锁表的方式(类似于Java多线程中的锁)使得其他的线程只能在锁外等待,所以平时选用何种隔离级别应该根据实际情况。InnoDB引擎默认的事务隔离级别为Repeatable read (可重复读)。
  InnoDB引擎支持上面四种隔离级别,默认的为Repeatable read (可重复读);而在Oracle数据库中,只支持Serializable (串行化)级别和Read committed (读已提交)这两种级别,其中默认的为Read committed级别。
 

参考文献

posted @ 2017-09-17 22:42  傍晚的羔羊  阅读(955)  评论(0编辑  收藏  举报