一个事务中 可以查询自己未提交的数据吗_浅析数据库事务隔离级别

567e757030837ba67751507a851b62aa.png

数据库的事务隔离级别共分为4种,级别越高性能越差,合理的设置事务隔离级别才能保证数据正确的情况下性能最好。

隔离级别没有合理的设置,会出现一些异常情况,比如脏读、不可重复读、幻读。根据自身的业务场景去分析是否允许这些异常情况的存在,设置合理的隔离级别

一、概念介绍

脏读

在一个事务中读取到了其他事务未提交的数据,就是脏读。未提交就可能出现回滚,如果数据被回滚,那么该事务对回滚的数据进行操作必然会导致数据错乱,所以脏读情况要尽可能避免

可重复读

在一个事务中,从开始到结束的任意一瞬间读取到的数据应该都是一致的。如果不一致,通常情况下是别的事务对这些数据进行了更新操作

幻读

幻读常见于对数据进行插入操作,比如事务A插入了一些数据还未来得及提交,事务B插入的相同条数的数据并提交,这时事务A进行提交时会发现自己插入的数据不是预期的结果,感觉像出现了幻觉一样

二、四种事务隔离级别

read-uncommitted

读未提交,该级别的事务允许读取其他事务未提交的数据。会出现脏读、不可重复读、幻读

read-committed

读已提交,该级别的事务只能读取到其他事务已提交的数据。避免了脏读,会出现不可重复读、幻读

repeatable-read

重复读,是mysql默认的隔离级别,该级别的事务只能读取到其他事务已提交的数据,并且在事务的开始到结束的整个过程,读取到的数据都是一致的。避免了脏读、不可重复读,会出现幻读

serializable

串行化,该级别的事务会对读取到数据进行锁定,其他事务想要访问这些数据只能等待,所以是串行,性能较差。避免了脏读、不可重复读、幻读

三、演示准备

使用命令行登录mysql,-u指定用户名;-P指定端口,mysql默认端口是3306,我的数据库使用的是3307;-p参数会提示输入密码

mysql -u root -P 3307 -p

01eacb875412fd9a798294e3650b4467.png

创建demo数据库,创建trans表,向表中添加两条数据;

create database demo;
use demo;
create table trans(id int(11) primary key,name varchar(50));
insert trans values (1,'张三');
insert trans values (2,'李四');
select * from trans;

af407e8d3801d3dedada2f248900cba8.png

查看并关闭当前会话的自动提交,1是开启自动提交,0是关闭自动提交

select @@autocommit;
set autocommit=0; 

ee5b379ec7d492ecfa8744e1c80303b4.png

打开另一个命令行窗口,登录mysql,并关闭自动提交

四、演示脏读

在A窗口查看当前会话的隔离级别,并设置为读未提交。查看表,可以看到之前初始化的两条数据

select @@tx_isolation;
set tx_isolation='read-uncommitted';

58fc8b765e62dcf7ba9eb07a79931a4e.png

在B窗口开启一个事务,更新id=1的名字为张三1

start transaction;
update trans set name='张三1' where id=1;

05098359490d96d58e86c00f9122eaa7.png

在A窗口查看表,可以看到B窗口未提交的数据被读取到了,这就是脏读

da3fe0504b84b2cd708d8333985e329d.png

将事务隔离级别提高即可避免出现脏读。为方便后面的演示,在B窗口中把数据回滚为最初状态

rollback;

0992193d46c6ecbf6ae53b76f91bc98b.png

五、演示不可重复读

在A窗口设置事务隔离级别为读已提交,然后开启事务,查询表中数据

set tx_isolation='read-committed';

e0a6d2aca81b80efee6d5977d82078dc.png

在B窗口更新数据,并提交

1d4d6d2e03d4df18380dea88157f99c6.png

在A窗口再次查看表,可以看到,同一个事务中两次的查询结果不一样,这就是不可重复读。假设在两次查询之间有更新操作,有可能造成更新的结果和预期的不一致

83cbf8c5d1eaf6946119c07f0d0365c4.png

将隔离级别提高为reapeatable-read可避免不可重复读,但级别越高性能越差。读已提交是常用的事务隔离级别,有些业务需要可重复读时,加上乐观锁即可

六、演示幻读

在A窗口开启事务,查询数据,可以看到两条数据

48b9358d01210a408c1de328e5c4f9dd.png

在窗口B插入一条数据,并提交

595aa9a325f2ecf2329b548f6a079560.png

假设窗口A并不知道B窗口插入了一条数据,A窗口也想插入这条数据就会报错

9d19dd63a55e148de1f901a9d3040f24.png

从A窗口的角度分析,在同一个事务中,刚开始查的时候没有id=3的这条记录,等到插入时报错主键冲突,感觉很魔幻,这就是幻读。

七、其他小知识点

@@在mysql中表示内置变量,内置变量可对当前会话生效或者全局生效。上文演示的所有内置变量是仅当前会话生效,也可以使用gloabl关键字使内置变量全局生效。

select @@gloabl.autocommit;
set @@global.autocommit=1;

http://weixin.qq.com/r/bTpCWnPE7i6RrTc5929- (二维码自动识别)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值