通过案例带你了解 MySQL中的事务隔离级别


本篇文章通过四个案例带你了解 mysql中的事务隔离级别,希望对大家有所帮助!

通过案例带你了解 MySQL中的事务隔离级别

很多小伙伴对 MySQL 的隔离级别一直心存疑惑,其实这个问题一点都不难,关键看怎么讲!单纯的看理论,绝对让你晕头转向,但是,如果我们通过几个实际的 SQL 来演示一些,大家就会发现这玩意原来这么简单!【相关推荐:mysql视频教程】

今天松哥想通过几个简单的案例,来和大家演示一下 MySQL 中的事务隔离级别问题。

1. 理论

MySQL 中事务的隔离级别一共分为四种,分别如下:

  • 序列化(SERIALIZABLE)
  • 可重复读(REPEATABLE READ)
  • 提交读(READ COMMITTED)
  • 未提交读(READ UNCOMMITTED)

四种不同的隔离级别含义分别如下:

  • SERIALIZABLE

如果隔离级别为序列化,则用户之间通过一个接一个顺序地执行当前的事务,这种隔离级别提供了事务之间最大限度的隔离。
  • REPEATABLE READ

在可重复读在这一隔离级别上,事务不会被看成是一个序列。不过,当前正在执行事务的变化仍然不能被外部看到,也就是说,如果用户在另外一个事务中执行同条 SELECT 语句数次,结果总是相同的。(因为正在执行的事务所产生的数据变化不能被外部看到)。

  • READ COMMITTED

READ COMMITTED 隔离级别的安全性比 REPEATABLE READ 隔离级别的安全性要差。处于 READ COMMITTED 级别的事务可以看到其他事务对数据的修改。也就是说,在事务处理期间,如果其他事务修改了相应的表,那么同一个事务的多个 SELECT 语句可能返回不同的结果。

  • READ UNCOMMITTED

READ UNCOMMITTED 提供了事务之间最小限度的隔离。除了容易产生虚幻的读操作和不能重复的读操作外,处于这个隔离级的事务可以读到其他事务还没有提交的数据,如果这个事务使用其他事务不提交的变化作为计算的基础,然后那些未提交的变化被它们的父事务撤销,这就导致了大量的数据变化。

在 MySQL 数据库种,默认的事务隔离级别是 REPEATABLE READ

2. SQL 实践

接下来通过几条简单的 SQL 向读者验证上面的理论。

2.1 查看隔离级别

通过如下 SQL 可以查看数据库实例默认的全局隔离级别和当前 session 的隔离级别:

MySQL8 之前使用如下命令查看 MySQL 隔离级别:

SELECT @@GLOBAL.tx_isolation, @@tx_isolation;

查询结果如图:

1.png

可以看到,默认的隔离级别为 REPEATABLE-READ,全局隔离级别和当前会话隔离级别皆是如此。

MySQL8 开始,通过如下命令查看 MySQL 默认隔离级别

SELECT @@GLOBAL.transaction_isolation, @@transaction_isolation;

就是关键字变了,其他都一样。

通过如下命令可以修改隔离级别(建议开发者在修改时修改当前 session 隔离级别即可,不用修改全局的隔离级别):

SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED

上面这条 SQL 表示将当前 session 的数据库隔离级别设置为 READ UNCOMMITTED,设置成功后,再次查询隔离级别,发现当前 session 的隔离级别已经变了,如图1-2:

2.png

注意,如果只是修改了当前 session 的隔离级别,则换一个 session 之后,隔离级别又会恢复到默认的隔离级别,所以我们测试时,修改当前 session 的隔离级别即可。

2.2 READ UNCOMMITTED

2.2.1 准备测试数据

READ UNCOMMITTED 是最低隔离级别,这种隔离级别中存在脏读、不可重复读以及幻象读问题,所以这里我们先来看这个隔离级别,借此大家可以搞懂这三个问题到底是怎么回事。

下面分别予以介绍。

首先创建一个简单的表,预设两条数据,如下:

3.png

表的数据很简单,有 j*aboy 和 itboyhub 两个用户,两个人的账户各有 1000 人民币。现在模拟这两个用户之间的一个转账操作。

注意,如果读者使用的是 N*icat 的话,不同的查询窗口就对应了不同的 session,如果读者使用了 SQLyog 的话,不同查询窗口对应同一个 session,因此如果使用 SQLyog,需要读者再开启一个新的连接,在新的连接中进行查询操作。

2.2.2 脏读

一个事务读到另外一个事务还没有提交的数据,称之为脏读。具体操作如下:

  • 首先打开两个SQL操作窗口,假设分别为 A 和 B,在 A 窗口中输入如下几条 SQL (输入完成后不用执行):

START TRANSACTION;
UPDATE account set balance=balance+100 where name='j*aboy';
UPDATE account set balance=balance-100 where name='itboyhub';COMMIT;
  • 在 B 窗口执行如下 SQL,修改默认的事务隔离级别为 READ UNCOMMITTED,如下:

SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED
  • 接下来在 B 窗口中输入如下 SQL,输入完成后,首先执行第一行开启事务(注意只需要执行一行即可):

START TRANSACTION;SELECT * from account;COMMIT;
  • 接下来执行 A 窗口中的前两条 SQL,即开启事务,给 j*aboy 这个账户添加 100 元。

  • 进入到 B 窗口,执行 B 窗口的第二条查询 SQL(SELECT * from user;),结果如下:

4.png

可以看到,A 窗口中的事务,虽然还未提交,但是 B 窗口中已经可以查询到数据的相关变化了。

这就是脏读问题。

2.2.3 不可重复读

不可重复读是指一个事务先后读取同一条记录,但两次读取的数据不同,称之为不可重复读。具体操作步骤如下(操作之前先将两个账户的钱都恢复为1000):

  1. 首先打开两个查询窗口 A 和 B ,并且将 B 的数据库事务隔离级别设置为 READ UNCOMMITTED。具体 SQL 参考上文,这里不赘述。

    SONIFY.io SONIFY.io

    设计和开发音频优先的产品和数据驱动的解决方案

    SONIFY.io 83 查看详情 SONIFY.io
  2. 在 B 窗口中输入如下 SQL,然后只执行前两条 SQL 开启事务并查询 j*aboy 的账户:

START TRANSACTION;SELECT * from account where name='j*aboy';COMMIT;

前两条 SQL 执行结果如下:

5.png

  1. 在 A 窗口中执行如下 SQL,给 j*aboy 这个账户添加 100 块钱,如下:
START TRANSACTION;
UPDATE account set balance=balance+100 where name='j*aboy';COMMIT;

4.再次回到 B 窗口,执行 B 窗口的第二条 SQL 查看 j*aboy 的账户,结果如下:

6.png

j*aboy 的账户已经发生了变化,即前后两次查看 j*aboy 账户,结果不一致,这就是不可重复读

和脏读的区别在于,脏读是看到了其他事务未提交的数据,而不可重复读是看到了其他事务已经提交的数据(由于当前 SQL 也是在事务中,因此有可能并不想看到其他事务已经提交的数据)。

2.2.4 幻象读

幻象读和不可重复读非常像,看名字就是产生幻觉了。

我举一个简单例子。

在 A 窗口中输入如下 SQL:

START TRANSACTION;insert into account(name,balance) values('zhangsan',1000);COMMIT;

然后在 B 窗口输入如下 SQL:

START TRANSACTION;SELECT * from account;delete from account where name='zhangsan';COMMIT;

我们执行步骤如下:

  • 首先执行 B 窗口的前两行,开启一个事务,同时查询数据库中的数据,此时查询到的数据只有 j*aboy 和 itboyhub。

  • 执行 A 窗口的前两行,向数据库中添加一个名为 zhangsan 的用户,注意不用提交事务。

  • 执行 B 窗口的第二行,由于脏读问题,此时可以查询到 zhangsan 这个用户。

  • 执行 B 窗口的第三行,去删除 name 为 zhangsan 的记录,这个时候删除就会出问题,虽然在 B 窗口中可以查询到 zhangsan,但是这条记录还没有提交,是因为脏读的原因才看到了,所以是没法删除的。此时就产生了幻觉,明明有个 zhangsan,却无法删除。

这就是幻读

看了上面的案例,大家应该明白了脏读不可重复读以及幻读各自是什么含义了。

2.3 READ COMMITTED

和 READ UNCOMMITTED 相比,READ COMMITTED 主要解决了脏读的问题,对于不可重复读和幻象读则未解决。

将事务的隔离级别改为 READ COMMITTED 之后,重复上面关于脏读案例的测试,发现已经不存在脏读问题了;重复上面关于不可重复读案例的测试,发现不可重复读问题依然存在。

上面那个案例不适用于幻读的测试,我们换一个幻读的测试案例。

还是两个窗口 A 和 B,将 B 窗口的隔离级别改为 READ COMMITTED

然后在 A 窗口输入如下测试 SQL:

START TRANSACTION;insert into account(name,balance) values('zhangsan',1000);COMMIT;

在 B 窗口输入如下测试 SQL:

START TRANSACTION;SELECT * from account;insert into account(name,balance) values('zhangsan',1000);COMMIT;

测试方式如下:

  • 首先执行 B 窗口的前两行 SQL,开启事务并查询数据,此时查到的只有 j*aboy 和 itboyhub 两个用户。

  • 执行 A 窗口的前两行 SQL,插入一条记录,但是并不提交事务。

  • 执行 B 窗口的第二行 SQL,由于现在已经没有了脏读问题,所以此时查不到 A 窗口中添加的数据。

  • 执行 B 窗口的第三行 SQL,由于 name 字段唯一,因此这里会无法插入。此时就产生幻觉了,明明没有 zhangsan 这个用户,却无法插入 zhangsan。

2.4 REPEATABLE READ

和 READ COMMITTED 相比,REPEATABLE READ 进一步解决了不可重复读的问题,但是幻象读则未解决。

REPEATABLE READ 中关于幻读的测试和上一小节基本一致,不同的是第二步中执行完插入 SQL 后记得提交事务。

由于 REPEATABLE READ 已经解决了不可重复读,因此第二步即使提交了事务,第三步也查不到已经提交的数据,第四步继续插入就会出错。

注意,REPEATABLE READ 也是 InnoDB 引擎的默认数据库事务隔离级别

2.5 SERIALIZABLE

SERIALIZABLE 提供了事务之间最大限度的隔离,在这种隔离级别中,事务一个接一个顺序的执行,不会发生脏读、不可重复读以及幻象读问题,最安全。

如果设置当前事务隔离级别为 SERIALIZABLE,那么此时开启其他事务时,就会阻塞,必须等当前事务提交了,其他事务才能开启成功,因此前面的脏读、不可重复读以及幻象读问题这里都不会发生。

3. 总结

总的来说,隔离级别和脏读、不可重复读以及幻象读的对应关系如下:

隔离级别 脏读 不可重复读 幻象读
READ UNCOMMITTED 允许 允许 允许
READ COMMITED 不允许 允许 允许
REPEATABLE READ 不允许 不允许 允许
SERIALIZABLE 不允许 不允许 不允许

性能关系如图:

7.png

好了,这篇文章就和小伙伴们先说这么多,大家不妨写几行 SQL 试一试。

更多编程相关知识,请访问:编程视频!!

以上就是通过案例带你了解 MySQL中的事务隔离级别的详细内容,更多请关注其它相关文章!


# 的是  # 景县衡水网站优化价格  # 盐城个人网站建设  # 广西seo厂家  # 成人网站建设建设  # 丝瓜seo130  # 项目的营销推广方案鱼鸭  # 地产营销推广费用侧算  # sem seo工资  # seo网站首页导航优化  # 德州专业网站优化哪家好  # MySQL  # 可以看到  # 两行  # 这就是  # 带你  # 还没有  # 两条  # 就会  # 窗口中  # 镜像  # 事务隔离级别 


相关栏目: 【 Google疑问12 】 【 Facebook疑问10 】 【 优化推广96088 】 【 技术知识133117 】 【 IDC资讯59369 】 【 网络运营7196 】 【 IT资讯61894


相关推荐: Flexbox布局实践:实现底部页脚与顶部粘性导航条的完美结合  《小黑盒》删除历史浏览方法  steam缓存文件在哪儿_steam缓存文件的路径查找方法与结构说明  全球各国上班时间表外贸邮件时间  KFC邀请码怎么使用领额外优惠_KFC邀请码输入方式与额外优惠代码获取方法  MySQL多重关联查询:利用别名高效获取同一表的多个关联字段  NumPy 高性能技巧:基于多列条件查找最近邻行索引的向量化实现  TikTok搜索结果不显示怎么办 TikTok搜索刷新与优化方法  excel怎么制作考勤表 excel考勤模板与函数公式讲解  如何快速去除厨房重油污? 2025年最好用的厨房清洁剂推荐  视频号视频怎么提取文案?提取的文案如何优化与使用?  网易云音乐闹钟铃声设置教程  51漫画网实时入口 51漫画网页版官方免费漫画入口  2025SNH48年度青春盛典门票价格及购买方式  谷歌邮箱怎么换绑定邮箱Gmail安全备份邮箱修改方法  J*aScript实现下拉菜单驱动的动态表格数据展示  第五人格PC版怎么避免被封号_第五人格PC版防封号注意事项  《三角洲行动》战斗步枪与机枪类改装代码分享  j*a中赋值运算符是什么?  12306售票时间最新规定 | 网上订票和车站窗口时间一样吗  win11如何开启单声道音频 Win11为听障用户合并左右声道【辅助】  AO3中文版手机快速通道_AO3最新稳定链接更新  Win10如何关闭开机锁屏界面_Windows10跳过锁屏直接登录设置  Lar*el 中高效执行多列更新:单次查询实现  J*a列表元素格式化输出教程  VS Code源代码管理(SCM)视图的进阶使用技巧  C++怎么实现一个红黑树_C++高级数据结构与平衡二叉搜索树  微博网页版访问入口 微博网页版网页端使用指南  悟空浏览器网页版在线工具 悟空浏览器网页版在线平台入口  QQ邮箱PC端登录页面_QQ邮箱网页版登录界面  猫眼电影app怎么查询电影院的营业时间_猫眼电影影院营业时间查询教程  win11自带录屏文件保存在哪里 Win11 Game Bar录制视频默认路径【分享】  太平年在哪个平台播出  《广发易淘金》国债逆回购操作教程  《sketchbook》选中部分图案移动方法  铁路12306买票怎么选双人铺 铁路12306卧铺分配规则说明  《王者荣耀世界》英雄获取攻略  秋风萧瑟洪波涌起中的萧瑟指的是什么  QQ邮箱官方登录页_腾讯出品安全稳定的邮箱服务  钉钉任务无法提醒如何处理 钉钉任务提醒优化方法  汽水音乐官网网页版入口 汽水音乐官网网页版在线入口  《浙里办》电子发票开具方法  Python中深度嵌套字典与列表的数据提取与条件过滤指南  抖音号已注销怎么解绑企业认证?不解绑企业认证会怎样?  解决SQLAlchemy模型跨文件关联的Linter兼容性指南  微信朋友圈怎么设置三天可见 微信朋友圈设置指定天数可见步骤【教程】  铁路12306官网入口 铁路12306中国铁路官网登录首页  b站如何管理订阅_b站订阅标签分类管理  XPath动态元素定位:如何精准选择文本内容变化的元素  大熊猫抓取竹子的“大拇指”其实是什么?蚂蚁庄园课堂今天答案最新11月30日 

 2021-09-23

了解您产品搜索量及市场趋势,制定营销计划

同行竞争及网站分析保障您的广告效果

点击免费数据支持

提交您的需求,1小时内享受我们的专业解答。

运城市盐湖区信雨科技有限公司


运城市盐湖区信雨科技有限公司

运城市盐湖区信雨科技有限公司是一家深耕海外推广领域十年的专业服务商,作为谷歌推广与Facebook广告全球合作伙伴,聚焦外贸企业出海痛点,以数字化营销为核心,提供一站式海外营销解决方案。公司凭借十年行业沉淀与平台官方资源加持,打破传统外贸获客壁垒,助力企业高效开拓全球市场,成为中小企业出海的可靠合作伙伴。

 8156699

 13765294890

 8156699@qq.com

Notice

We and selected third parties use cookies or similar technologies for technical purposes and, with your consent, for other purposes as specified in the cookie policy.
You can consent to the use of such technologies by closing this notice, by interacting with any link or button outside of this notice or by continuing to browse otherwise.