浅析数据库事务隔离级别

Python涨薪研究所

共 2117字,需浏览 5分钟

 ·

2021-10-03 16:06


源 /         文/ 王小伍


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



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


01

概念介绍

脏读

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



可重复读

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



幻读

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


02

四种事务隔离级别

read-uncommitted

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



read-committed

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



repeatable-read

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



serializable

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


03

演示准备

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

mysql -u root -P 3307 -p



创建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;



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

select @@autocommit;set autocommit=0;



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


04

演示脏读

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

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



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

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



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



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

rollback;


05

演示不可重复读

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

set tx_isolation='read-committed';



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



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



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


06

演示幻读

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



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



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



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


07

其他小知识点

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

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



 推荐阅读


华为最美小姐姐被外派墨西哥后...


国内有程序员电视剧了,结果看了一分钟,就吐了...


男女洗澡前后区别,太形象了!















END


顶级程序员:topcoding

做最好的程序员社区:Java后端开发、Python、大数据、AI


一键三连「分享」、「点赞」和「在看」


浏览 23
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报