vlambda博客
学习文章列表

MySQL事务,不成功便成仁

一、什么是事务 
    一个事务是由一条或者多条sql语句构成,这一条或者多条sql语句要么全部执行成功,要么全部执行失败!
    MySQL的事务支持是在存储引擎层面实现的。InnoDB支持事务,而MyISAM不支持。
二、事务的四大特性(ACID)
  • 原子性(Atomicity):事务中所有操作是不可再分割的原子单位。事务中所有操作要么全部执行成功,要么全部执行失败。
  • 一致性(Consistency):事务执行后,数据库状态与其它业务规则保持一致。如转账业务,无论事务执行成功与否,参与转账的两个账号余额之和应该是不变的。
  • 隔离性(Isolation):隔离性是指在并发操作中,不同事务之间应该隔离开来,使每个并发中的事务不会相互干扰。
  • 持久性(Durability):一旦事务提交成功,事务中所有的数据操作都必须被持久化到数据库中,即使提交事务后,数据库马上崩溃,在数据库重启时,也必须能保证通过某种机制恢复数据。
三、事务的开启和结束
    在默认情况下,MySQL每执行一条SQL语句,都是一个单独的事务。如果需要在一个事务中包含多条SQL语句,那么需要开启事务和结束事务。
  1. set autocommit=1(MySQL默认配置)需要显式启动事务语句:begin 或 start transaction。配套的提交语句是 commit,回滚语句是 rollback。如果提交一个事务后要自动开启下一个事务,可以使用commit work and chain来提交上一个事务并开启下一个事务。
  2. set autocommit=0:这个命令会将这个线程的自动提交关掉。意味着你只执行一个 select 语句,这个事务就启动了,而且并不会自动提交。这个事务持续存在直到你主动执行 commit 或 rollback 语句,或者断开连接。
    最后要结束事务,commit表示提交,即事务中的多条SQL语句所做出的影响会持久化到数据库中。或者rollback,表示回滚,即回滚到事务的起点,之前做的所有操作都被撤消了!
四、事务的并发问题
  • 脏读:一个事务中读取了另一个未提交事务中的数据

    如:A向B的账户执行了两条操作,+100元和-100元,两条操作是一个事务;当执行完+100时,B查看账户发现多了100(此时发生了脏读),认为A已经向自己打了100,但事务执行完毕结果是没有打钱,当B再次查看账户时发现钱没有多。

  • 不可重复读:一个事务中多次访问某个数据,而访问过程中这个数据被其他事务做了修改并提交成功,这样前后访问这个数据时就得到了不同的值
  • 幻读:幻读是事务非独立执行时发生的一种现象。

    如:事务T1对一个表中所有的行的某个数据项做了从“1”修改为“2”的操作,这时事务T2又对这个表中插入了一行数据项,而这个数据项的数值还是为“1”并且提交给数据库。而操作事务T1的用户如果再查看刚刚修改的数据,会发现还有一行没有修改,其实这行是从事务T2中添加的,就好像产生幻觉一样,这就是发生了幻读

不可重复读和脏读的区别 :脏读是读到了其他未提交事务的数据,而不可重复的是读到了其他已提交事务的数据。
幻读和不可重复读的区别 是读取了另一 已经提交的事务(这点就脏读不同),所不同的是不可重复读查询的都是同一个数据项,而幻读针对的是一批数据整体(比如数据的个数)。
五、MySQL事务的隔离级别
  • Serializable (串行化):可避免脏读、不可重复读、幻读的发生。(添加表锁,效率低)
  • Repeatable read (可重复读,默认):可避免脏读、不可重复读的发生。
  • Read committed (读已提交):可避免脏读的发生。
  • Read uncommitted (读未提交):最低级别,无法避免任何并发问题。
    注意: 隔离级别的设置只对当前连接有效 。对于使用MySQL命令窗口而言,一个窗口就相当于一个连接;当前窗口设置的隔离级别只对当前窗口中的事务有效;对于JDBC操作数据库来说,一个Connection对象相当于一个连接,而对于Connection对象设置的隔离级别只对该Connection对象有效,与其他连接Connection对象无关。
查看当前连接的隔离级别:

5.6版本及之前:

show variables like 'tx_isolation'

5.7版本之后

show variables like 'transaction_isolation'

举例说明4种隔离级别

mysql> create table test(id int) engine=InnoDB;insert into test(id) values(1);

  • 若隔离级别是“读未提交”, 则 V1 的值就是 2。这时候事务 B 虽然还没有提交,但是结果已经被 A 看到了。因此,V2、V3 也都是 2。
  • 若隔离级别是“读提交”,则 V1 是 1,V2 的值是 2。事务 B 的更新在提交后才能被 A 看到。所以, V3 的值也是 2。
  • 若隔离级别是“可重复读”,则 V1、V2 是 1,V3 是 2。之所以 V2 还是 1,遵循的就是这个要求:事务在执行期间看到的数据前后必须是一致的。
  • 若隔离级别是“串行化”,则在事务 B 执行“将 1 改成 2”的时候,会被锁住。直到事务 A 提交后,事务 B 才可以继续执行。所以从 A 的角度看, V1、V2 值是 1,V3 的值是 2。
事务隔离性的实现方式
    在实现上,数据库里面会创建一个视图,访问的时候以视图的逻辑结果为准。
  • “可重复读”隔离级别下,这个视图是在 事务启动时创建的 ,整个事务存在期间都用这个视图。
  • “读提交”隔离级别 下,这个视图是在 每个 SQL 语句开始执行的时候创建的
  • “读未提交”隔离级别下直接返回记录上的最新值,没有视图概念。
  • “串行化”隔离级别下直接用加锁的方式来避免并行访问。