vlambda博客
学习文章列表

浅析数据库事务隔离级别

👇👇 关注后回复 “进群” ,拉你进程序员交流群 👇👇
架构师大咖
架构师大咖,打造有价值的架构师交流平台。分享架构师干货、教程、课程、资讯。架构师大咖,每日推送。
0篇原创内容
Official Account
浅析数据库事务隔离级别
算法专栏
算法专栏,每日推送。算法是程序员内功,分享算法知识、文章、工具、算法题、教程等
0篇原创内容
Official Account

作者丨王小伍

来源丨赫连小伍



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

最近有一些小伙伴,让我帮忙找一些 面试题 资料,于是我翻遍了收藏的 5T 资料后,汇总整理出来,可以说是程序员面试必备!所有资料都整理到网盘了,欢迎下载!

浅析数据库事务隔离级别

程序员直聘
程序员直聘,一个程序员找工作平台。
21篇原创内容
Official Account
点击👆卡片,关注后回复【面试题】即可获取

在看点这里好文分享给更多人↓↓