【事务隔离级别】——深度探索数据库的事务隔离级别

  设置事务隔离级别的方式有很多种,上篇文章中只简单提到了理论知识,这里数据库以MySQL为例,来动手设置一下事务的隔离级别并观察一下事务的隔离级别到底起到什么作用。

  我们都知道,每启动一下MySQL,就会获得一个数据库连接,每个数据库连接有一个全局变量@@tx_isolation,表示当前连接中事务的隔离级别。

  查看事务隔离级别

  命令行登录mysql,查看当前事务隔离级别:

 select @@tx_isolation;

  或者

select @@session.tx_isolation;

【事务隔离级别】——深度探索数据库的事务隔离级别_第1张图片

  可以看到mysql默认的事务隔离级别为REPEATABLE-READ

  查看系统全局事务离级别可以用:

select @@global.tx_isolation;



  设置事务隔离级别

  设置当前事务隔离级别

set session transaction isolation level repeatable read;

  设置系统全局(默认)事务隔离级别

set global transaction isolation level repeatable read;



  read uncommitted测试

  打开一个客户端A,并设置当前事务模式为read uncommitted(未提交读),查询表account的初始值:

【事务隔离级别】——深度探索数据库的事务隔离级别_第2张图片

  在客户端A的事务提交之前,打开另一个客户端B,更新表account:

【事务隔离级别】——深度探索数据库的事务隔离级别_第3张图片

  这时,虽然客户端B的事务还没提交,但是客户端A就可以查询到B已经更新的数据:

【事务隔离级别】——深度探索数据库的事务隔离级别_第4张图片

  一旦客户端B的事务因为某种原因回滚,所有的操作都将会被撤销,那客户端A查询到的数据其实就是脏数据:

【事务隔离级别】——深度探索数据库的事务隔离级别_第5张图片



  read committed测试

  针对上面的问题,把客户端A的事务隔离级别设置为read committed,再重复上面的步骤,会发现B在事务结束之前,A并不能查询到B所做的操作。

  客户端A的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第6张图片

  客户端B的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第7张图片

  但是正如上文所说,这种隔离级别下可能导致前事务中多次读取特定记录的结果不相同,比如客户端A事务隔离级别为read committed,在A的一个事务中,执行两次相同的查询,在这两次查询的中间,客户端B对数据进行更改并提交事务,那么会导致客户端A的两次查询结果不一致,导致“不可重复读”的麻烦。

  客户端A的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第8张图片

  客户端B的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第9张图片



    repeatable read测试

  同理,如果设置了repeatable read隔离级别,就可以保证在当前事务中多次执行相同查询的结果集相同,实现“可重复读”。

  同样以客户端A设置事务隔离级别为repeatable read为例,执行结果如下:

  客户端A的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第10张图片

  客户端B的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第11张图片

  上篇文章说到,这种隔离级别会导致“幻读”,比如客户端A中事务操作表中符合条件的若干行,同时客户端B中事务插入符合A操作条件的数据行,然后再提交。究竟会发生什么nie~~

  小编为此也进行了测试:

  客户端A的执行结果
【事务隔离级别】——深度探索数据库的事务隔离级别_第12张图片

  客户端B的执行结果

【事务隔离级别】——深度探索数据库的事务隔离级别_第13张图片

  结果却不像我们预测的那样,为客户端A中的事务设置隔离级别为repeatable read,但在客户端B中的事务插入数据后,A并没有出现“幻读”的现象。查了资料才知道,原来在mysql中,不会出现幻读。mysql的实现和标准定义的RR隔离级别有差别。

当然,
  上篇文章说到的只是SQL规范所规定的标准,不同的数据库具体的实现可能会有些差异,所以还是以实际为准~~
  实践出真知,所以每接触到一些理论知识时,最好能够实践一把,自己动手去探索,总会带给你意想不到的收货哦。

你可能感兴趣的:(mysql,数据库,事务,事务隔离级别)