MySQL---事务+事务隔离级别

一、事务定义

 

  • Transaction
  • 事务:一个最小的不可再分的工作单元;通常一个事务对应一个完整的业务(例如银行账户转账业务,该业务就是一个最小的工作单元)
  • 一个完整的业务需要批量的DML(insert、update、delete)语句共同联合完成
  • 事务只和DML语句有关,或者说DML语句才有事务。这个和业务逻辑有关,业务逻辑不同,DML语句的个数不同



二、转账操作理解事务


关于银行账户转账操作,账户转账是一个完整的业务,最小的单元,不可再分—也就是说银行账户转账是一个事务



以下是银行账户表t_act(账号、余额),进行转账操作

actno		balance
1			500
2			100


转账操作

update t_act set balance=400 where actno=1;
update t_act set balance=200 where actno=2;

以上两台DML语句必须同时成功或者同时失败。最小单元不可再分,当第一条DML语句执行成功后,并不能将底层数据库中的第一个账户的数据修改,只是将操作记录了一下;这个记录是在内存中完成的;当第二条DML语句执行成功后,和底层数据库文件中的数据完成同步。若第二条DML语句执行失败,则清空所有的历史操作记录,要完成以上的功能必须借助事务




三、事务四大特征(ACID)

  • 原子性(A):事务是最小单位,不可再分
  • 一致性©:事务要求所有的DML语句操作的时候,必须保证同时成功或者同时失败
  • 隔离性(I):并发时,事务A和事务B之间具有隔离性
  • 持久性(D):是事务的保证,事务终结的标志(内存的数据持久到硬盘文件中)


五、和事务相关的两条重要的SQL语句(TCL)

  • commit:提交
  • rollback:回滚         (事务commit之前可以rollback,commit之后rollback就无效了)


七、事务与数据库底层数据

在事务进行过程中,未结束之前,DML语句是不会更改底层数据,只是将历史操作记录一下,在内存中完成记录。只有在事物结束的时候,而且是成功的结束的时候,才会修改底层硬盘文件中的数据




八、在MySQL中,事务提交与回滚

在MySQL中,默认情况下,事务是自动提交的,自动提交机制是可以关闭的:

方式一:
start transaction;  //手动开启事务
DML语句...
commit;   //手动提交事务(事务成功地结束)

start transaction;  //手动开启事务
DML语句...
rollback;   //手动回滚事务(事务失败地结束)

方式二:
关闭自动提交:
set autocommit=off;
或:set session autocommit=off;
打开自动提交
set autocommit=on;
或:set session autocommit=on;
以上打开和关闭自动提交机制,只对当前会话有效

对 t_user 进行提交和回滚操作

提交操作(事务成功)

  • start transaction

  • DML语句

  • commit

      mysql> start transaction;  #手动开启事务,mysql命令行输入这个后,
                                 #就关闭了自动提交功能,需要手动提交和回滚
      mysql> insert into t_user(name) values('pp');
      mysql> commit;             #commit之后即可改变底层数据库数据
      mysql> select * from t_user;
      +----+------+
      | id | name |
      +----+------+
      |  1 | jay  |
      |  2 | man  |
      |  3 | pp   |
      +----+------+
      3 rows in set (0.00 sec)
    


回滚操作(事务失败)

  • start transaction

  • DML语句

  • rollback

      mysql> start transaction;
      mysql> insert into t_user(name) values('yy');
      mysql> rollback;
      mysql> select * from t_user;
      +----+------+
      | id | name |
      +----+------+
      |  1 | jay  |
      |  2 | man  |
      |  3 | pp   |
      +----+------+
      3 rows in set (0.00 sec)
    



九、事务四大特性之一————隔离性(isolation)

MySql默认是隔离级别是可重复读(历史原因造成),Oracle,SqlServer 的默认事务隔离级别是读已提交(Read Commited);

一般互联网项目中mysql的事务隔离级别设置成 Read Commited 读已提交,因为实际工作中不需要可重复读,可重复读有多缺点(死锁概率高;在RR隔离级别下,条件列未命中索引会锁表!而在RC隔离级别下,只锁行;等等)参考:互联网项目中mysql应该选什么事务隔离级别 转(示例代码)_136.la

Mysql的四种隔离级别

SQL标准定义了4类隔离级别,包括了一些具体规则,用来限定事务内外的哪些改变是可见的,
哪些是不可见的。低级别的隔离级一般支持更高的并发处理,并拥有更低的系统开销。


Read Uncommitted(读取未提交内容)  ru
在该隔离级别,所有事务都可以看到其他未提交事务的执行结果。本隔离级别很少用于实际应用,
因为它的性能也不比其他级别好多少。读取未提交的数据,也被称之为脏读(Dirty Read)。


Read Committed(读取提交内容)     rc
这是大多数数据库系统的默认隔离级别(但不是MySQL默认的)。它满足了隔离的简单定义:
一个事务只能看见已经提交事务所做的改变。
这种隔离级别 也支持所谓的不可重复读(Nonrepeatable Read),
因为同一事务的其他实例在该实例处理其间可能会有新的commit,所以同一select可能返回不同结果。

  
Repeatable Read(可重读)         rr
这是MySQL的默认事务隔离级别,它确保同一事务的多个实例在并发读取数据时,会看到同样的数据行。
不过理论上,这会导致另一个棘手的问题:幻读 (Phantom Read)。
简单的说,幻读指当用户读取某一范围的数据行时,另一个事务又在该范围内插入了新行,
当用户再读取该范围的数据行时,会发现有新的“幻影” 行。
InnoDB和Falcon存储引擎通过多版本并发控制(MVCC,Multiversion Concurrency Control)机制
解决了该问题。


Serializable(可串行化)         s
这是最高的隔离级别,它通过强制事务排序,使之不可能相互冲突,从而解决幻读问题。
简言之,它是在每个读的数据行上加上共享锁。在这个级别,可能导致大量的超时现象和锁竞争。

隔离级别的不同会导致出现不同的问题:

这四种隔离级别采取不同的锁类型来实现,若读取的是同一个数据的话,就容易发生问题。例如:

脏读(Drity Read):某个事务已更新一份数据,另一个事务在此时读取了同一份数据,由于某些原因,前一个RollBack了操作,则后一个事务所读取的数据就会是不正确的。

不可重复读(Non-repeatable read):在一个事务的两次查询之中数据不一致,这可能是两次查询过程中间插入了一个事务更新的原有的数据。

幻读(Phantom Read):在一个事务的两次查询中数据笔数不一致,例如有一个事务查询了几列(Row)数据,而另一个事务却在此时插入了新的几列数据,先前的事务在接下来的查询中,就会发现有几列数据是它先前所没有的。

测试Mysql的隔离级别:

下面,将利用MySQL的客户端程序,我们分别来测试一下这几种隔离级别,两个命令行客户端分别为A,B;不断改变A的隔离级别,在B端修改数据。

Read Uncommitted(读取未提交内容):

(一)、将A的隔离级别设置为read uncommitted(未提交读)

set session transaction isolation level read uncommitted;

查看隔离级别是否设置成功

select @@transaction_isolation (mysql版本 8.0 以后)

select @@tx_isolation (mysql版本 8.0 之前)

查看mysql版本 

> status

 

A:启动事务,此时select读取数据为初始状态

 start transaction;

B:启动事务,更新数据,但不提交

 start transaction;

A:再次读取数据,发现数据已经被修改了,这就是所谓的“脏读”


B:回滚事务

rollback;
 
A:再次读数据,发现数据变回初始状态

经过上面的实验可以得出结论,事务B更新了一条记录,但是没有提交,
此时事务A可以查询出未提交记录。造成脏读现象。未提交读是最低的隔离级别。

 Read Committed(读取提交内容):

(二)、将客户端A的事务隔离级别设置为read committed(已提交读)

 set session transaction isolation level read committed;

A:启动事务,此时select读取数据为初始状态

B:启动事务,更新数据,但不提交

A:再次读数据,发现数据未被修改

B:提交事务

A:再次读取数据,发现数据已发生变化,说明B提交的修改被事务中的A读到了,
这就是所谓的“不可重复读”

 

经过上面的实验可以得出结论,已提交读隔离级别解决了脏读的问题,但是出现了不可重复读的问题,
即事务A在两次查询的数据不一致,
因为在两次查询之间事务B更新了一条数据。已提交读只允许读取已提交的记录,
但不要求可重复读。

Repeatable Read(可重读):

(三)、将A的隔离级别设置为repeatable read(可重复读)

A:启动事务,此时select读取数据为初始状态

B:启动事务,更新数据,但不提交

A:再次读取数据,发现数据未被修改

B:提交事务

A:再次读取数据,发现数据依然未发生变化,这说明这次可以重复读了

B:插入一条新的数据,并提交

A:再次读取数据,发现数据依然未发生变化,虽然可以重复读了,
但是却发现读的不是最新数据,这就是所谓的“幻读”

A:提交本次事务,再次读取数据,发现读取正常了

由以上的实验可以得出结论,可重复读隔离级别只允许读取已提交记录,
而且在一个事务两次读取一个记录期间,其他事务部的更新该记录。
但该事务不要求与其他事务可串行化。例如,当一个事务可以找到由一个已提交事务更新的记录,
但是可能产生幻读问题(注意是可能,因为数据库对隔离级别的实现有所差别)。
像以上的实验,就没有出现数据幻读的问题

Serializable(可串行化):

(四)、将A的隔离级别设置为可串行化(Serializable)

A:启动事务,此时数据为初始状态

B:发现B此时进入了等待状态,原因是因为A的事务尚未提交,只能等待(此时,B可能会发生等待超时)

A:提交事务

B:发现插入成功

serializable完全锁定字段,若一个事务来查询同一份数据就必须等待,
直到前一个事务完成并解除锁定为止。是完整的隔离级别,会锁定对应的数据表格,因而会有效率的问题。




十、隔离级别与一致性关系



这里写图片描述



十一、设置事务隔离级别



方式一

  • 可以在my.ini文件中使用transaction-isolation选项来设置服务器的缺省事务隔离级别。

    • 该选项值可以是:

        – READ-UNCOMMITTED
        – READ-COMMITTED
        – REPEATABLE-READ
        – SERIALIZABLE
      
        •	例如:
        [mysqld]
        transaction-isolation = READ-COMMITTED
      



方式二

  • 通过命令动态设置隔离级别
    • 隔离级别也可以在运行的服务器中动态设置,应使用SET TRANSACTION ISOLATION LEVEL语句。
    • 其语法模式为:

      		SET [GLOBAL | SESSION] TRANSACTION ISOLATION LEVEL <isolation-level>
      		其中的<isolation-level>可以是:
      	–	READ UNCOMMITTED
      	–	READ COMMITTED
      	–	REPEATABLE READ
      	–	SERIALIZABLE
      	•	例如: SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;
    



十二、隔离级别的作用范围

•	事务隔离级别的作用范围分为两种: 
–	全局级:对所有的会话有效 
–	会话级:只对当前的会话有效 
•	例如,设置会话级隔离级别为READ COMMITTED :
mysql> SET TRANSACTION ISOLATION LEVEL READ COMMITTED;
或:
mysql> SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
•	设置全局级隔离级别为READ COMMITTED : 
mysql> SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;



十三、查看隔离级别

•	事务隔离级别的作用范围分为两种: 
–	全局级:对所有的会话有效 
–	会话级:只对当前的会话有效 
•	例如,设置会话级隔离级别为READ COMMITTED :
mysql> SET TRANSACTION ISOLATION LEVEL READ COMMITTED;
或:
mysql> SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;
•	设置全局级隔离级别为READ COMMITTED : 
mysql> SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;

题外话:

事务是数据库 DML 语句中的,有 ACID 原则,数据库行锁是并发操作同一数据时防止数据混乱的。

悲观锁(行锁)、乐观锁、JUC的锁区分详见这2个博客:

MySql悲观锁(行锁)和乐观锁_逮虾户<的博客-CSDN博客

MySQL行锁详解(共享锁和排他锁)及应用级JUC锁和数据库锁的区别_逮虾户<的博客-CSDN博客

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值