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