MySQL事务特性及四种隔离级别
-
原子性(Atomicity):事务中所有操作是不可再分割的原子单位。事务中所有操作要么全部执行成功,要么全部执行失败。 -
一致性(Consistency):事务执行后,数据库状态与其它业务规则保持一致。如转账业务,无论事务执行成功与否,参与转账的两个账号余额之和应该是不变的。 -
隔离性(Isolation):隔离性是指在并发操作中,不同事务之间应该隔离开来,使每个并发中的事务不会相互干扰。 -
持久性(Durability):一旦事务提交成功,事务中所有的数据操作都必须被持久化到数据库中,即使提交事务后,数据库马上崩溃,在数据库重启时,也必须能保证通过某种机制恢复数据。
-
set autocommit=1(MySQL默认配置)需要显式启动事务语句:begin 或 start transaction。配套的提交语句是 commit,回滚语句是 rollback。如果提交一个事务后要自动开启下一个事务,可以使用commit work and chain来提交上一个事务并开启下一个事务。 -
set autocommit=0:这个命令会将这个线程的自动提交关掉。意味着你只执行一个 select 语句,这个事务就启动了,而且并不会自动提交。这个事务持续存在直到你主动执行 commit 或 rollback 语句,或者断开连接。
脏读:一个事务中读取了另一个未提交事务中的数据
如:A向B的账户执行了两条操作,+100元和-100元,两条操作是一个事务;当执行完+100时,B查看账户发现多了100(此时发生了脏读),认为A已经向自己打了100,但事务执行完毕结果是没有打钱,当B再次查看账户时发现钱没有多。
-
不可重复读:一个事务中多次访问某个数据,而访问过程中这个数据被其他事务做了修改并提交成功,这样前后访问这个数据时就得到了不同的值 幻读:幻读是事务非独立执行时发生的一种现象。
如:事务T1对一个表中所有的行的某个数据项做了从“1”修改为“2”的操作,这时事务T2又对这个表中插入了一行数据项,而这个数据项的数值还是为“1”并且提交给数据库。而操作事务T1的用户如果再查看刚刚修改的数据,会发现还有一行没有修改,其实这行是从事务T2中添加的,就好像产生幻觉一样,这就是发生了幻读
-
Serializable (串行化):可避免脏读、不可重复读、幻读的发生。(添加表锁,效率低) -
Repeatable read (可重复读,默认):可避免脏读、不可重复读的发生。 -
Read committed (读已提交):可避免脏读的发生。 -
Read uncommitted (读未提交):最低级别,无法避免任何并发问题。
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 语句开始执行的时候创建的 。 -
“读未提交”隔离级别下直接返回记录上的最新值,没有视图概念。 -
“串行化”隔离级别下直接用加锁的方式来避免并行访问。