浅析数据库事务隔离级别

数据库的事务隔离级别共分为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


创建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,并关闭自动提交


四、演示脏读

在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;


五、演示不可重复读

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

set tx_isolation='read-committed';


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


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


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


六、演示幻读

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


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


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


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


七、其他小知识点

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

select @@gloabl.autocommit;

set @@global.autocommit=1;


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