【MySQL】黑悟空都掌握的技能,数据库隔离级别全攻略

news/2024/9/14 2:16:28/ 标签: 数据库, mysql, sql

前言

🍊缘由

黑神话悟空玩家必备,数据库隔离级别完全解读

在这里插入图片描述

🐣闪亮主角

大家好,我是JavaDog程序狗

今天借着黑神话悟空的热度,跟大家分享一下数据库隔离级别,也是面试必备的八股文

在这里插入图片描述

😈你想听的故事

最近狗哥这当面试官的频率越来越多,面试者的水平也参差不齐

关于数据库隔离级别的问题也是五花八门,有的小伙伴说这完全是八股文,没有实际作用

但狗哥作为搬砖编码小王子,认为基础内容必须掌握,不只为了应付面试,而是提升自己的必经道路

遂狗哥总结数据库隔离级别全攻略,与小伙伴一起分享这个知识点

在这里插入图片描述

正文

🎯主要目标

1.MySQL事务隔离级别
2.如何设置隔离级别
3.实操分析隔离级别

🍪目标讲解

一.MySQL四大隔离级别
事务隔离级别
1.读未提交 (Read Uncommitted)
  • 允许一个事务读取另一个事务未提交的数据。
2.读已提交 (Read Committed)
  • 一个事务只能读取另一个事务已经提交的数据。
3.可重复读 (Repeatable Read)
  • 保证在一个事务内多次读取同一数据时,其结果是一致的。
  • 这是 MySQL 默认的隔离级别。
4.串行化 (Serializable)
  • 强制事务串行执行,避免所有并发问题。

在这里插入图片描述

隔离水平
1.读未提交 (Read Uncommitted)
  • 隔离水平:最低
  • 特性:允许一个事务读取另一个事务未提交的数据。
2.读已提交 (Read Committed)
  • 隔离水平:较低
  • 特性:一个事务只能读取另一个事务已经提交的数据。
3.可重复读 (Repeatable Read)
  • 隔离水平:较高
  • 特性:保证在一个事务内多次读取同一数据时,其结果是一致的。
  • 默认设置:这是 MySQL 默认的隔离级别。
4.串行化 (Serializable)
  • 隔离水平:最高
  • 特性:强制事务串行执行,避免所有并发问题。

在这里插入图片描述

产生问题
1. 读未提交 (Read Uncommitted)
  • 脏读:一个事务读取了另一个事务未提交的数据,如果之后该事务被回滚,则会导致读取的数据无效。
  • 幻读:一个事务读取了一组数据后,另一个事务插入了一些新的数据项,当第一个事务再次读取相同的数据集时,会发现多了未曾见过的行。
  • 不可重复读:一个事务读取了一条数据后,另一个事务修改了这条数据,当第一个事务再次读取时,得到的数据不同。
2. 读已提交 (Read Committed)
  • 不可重复读:一个事务读取了一条数据后,另一个事务修改并提交了这条数据,当第一个事务再次读取时,得到的数据不同。
  • 幻读:一个事务读取了一组数据后,另一个事务插入了一些新的数据项并提交,当第一个事务再次读取相同的数据集时,会发现多了未曾见过的行。
3. 可重复读 (Repeatable Read)
  • 幻读:一个事务读取了一组数据后,另一个事务插入了一些新的数据项并提交,当第一个事务再次读取相同的数据集时,会发现多了未曾见过的行。
    默认设置:这是 MySQL 默认的隔离级别。
4. 串行化 (Serializable)
  • 性能影响:由于所有事务必须串行执行,可能会导致性能下降。
隔离级别脏读幻读不可重复度
读未提交
读已提交不会
可重复读不会不会
串行化不会不会不会

🎯名词解释

  • 脏读
    定义:一个事务能够读取到另一个事务尚未提交的数据。

🌰现实例子:想象一下你在银行排队等待办理业务,而你的朋友正在柜台办理转账。你的朋友还没有完成转账操作,也就是说这笔钱还没有真正从他的账户转到你的账户。但是,你却提前看到了自己的账户余额增加了这笔钱。如果这时你的朋友决定取消转账,那么你看到的余额增加就是“脏”的,因为你实际上并没有收到这笔钱。

  • 幻读
    定义:幻读是指在一个事务内多次执行相同的查询语句,第二次或以后的查询返回了第一次查询时不存在的额外记录。

🌰现实例子:想象你在一个图书馆查找书籍,第一次查询时找到了10本书关于编程的书。你离开图书馆去喝了杯咖啡,回来后再次查询,发现现在有12本关于编程的书,因为在这段时间里有人捐赠了两本新书。这就像是在同一个事务中两次查询相同条件下的图书数量,但得到的数量不同。

  • 不可重复读
    定义:不可重复读是指在一个事务内多次执行相同的查询语句,第二次或以后的查询返回的结果与第一次查询的结果不同。

🌰现实例子:假设你在网上购物,第一次查看某商品的价格是100元。然后你去干别的事情,比如接了个电话。当你再次查看同一件商品的价格时,发现它变成了90元,因为在这段时间里商家做了促销活动降低了价格。这就像是在一个事务中两次查询同一商品的价格,但得到了不同的结果。


二.MySQL设置隔离级别
1.全局设置:
  • 修改 MySQL 的配置文件 (my.cnf 或 my.ini) 中的 innodb_locks_unsafe_for_binlog 和 transaction_isolation 参数
  • 这种设置适用于所有连接到 MySQL 的会话,并且需要重启 MySQL 服务才能生效。
2.会话级别设置:
  • 使用SET SESSION 或 SET 命令来设置当前会话的隔离级别
  • 这种设置只对当前会话有效,并且不会影响其他会话。

具体的 SQL 命令如下:

  • 设置全局隔离级别:
sql">  -- 修改配置文件中的设置# 在 my.cnf 或 my.ini 文件中添加如下设置[sql>mysqld]transaction_isolation = <isolation_level>
  • 设置当前会话隔离级别:
sql">  -- 设置当前会话的隔离级别为 REPEATABLE READSET SESSION TRANSACTION ISOLATION LEVEL REPEATABLE READ;-- 或者简写形式SET SESSION tx_isolation='REPEATABLE-READ';
  • 设置当前事务的隔离级别:
sql">  -- 在事务开始之前设置隔离级别SET TRANSACTION ISOLATION LEVEL REPEATABLE READ;

其中 <isolation_level> 是你想要设置的隔离级别,可以是以下之一:
READ UNCOMMITTED:最低的隔离级别,允许脏读。
READ COMMITTED:允许不可重复读。
REPEATABLE READ:MySQL 默认的隔离级别,防止不可重复读。
SERIALIZABLE:最高的隔离级别,完全防止脏读、不可重复读和幻读。

🌰示例
假设你想要设置当前会话的隔离级别为 READ COMMITTED,你可以这样操作:

sql">-- 查看当前会话的隔离级别
SELECT @@SESSION.transaction_isolation;-- 设置当前会话的隔离级别为 READ COMMITTED
SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED;-- 再次查看确认是否设置成功
SELECT @@SESSION.transaction_isolation;

💥注意:这些设置只会影响当前会话或当前事务,并且在事务结束后,隔离级别会恢复到之前的设置。如果你希望永久地更改隔离级别,你需要修改配置文件并重启 MySQL 服务。

三.实操分析隔离级别

在这里插入图片描述
🍓前置准备:

新建一个产品表,包含主键(id),价格(price),并插入一条数据,用作实际测试

sql">-- 创建表
CREATE TABLE `demo`.`Untitled`  (`id` bigint(20) NOT NULL COMMENT '主键id',`price` decimal(10, 2) NULL COMMENT '价格',PRIMARY KEY (`id`)
);
-- 测试数据
INSERT INTO `demo`.`product` (`id`, `price`) VALUES (1, 10.00);
INSERT INTO `demo`.`product` (`id`, `price`) VALUES (2, 10.00);
INSERT INTO `demo`.`product` (`id`, `price`) VALUES (3, 10.00);
INSERT INTO `demo`.`product` (`id`, `price`) VALUES (4, 10.00);
1.读未提交 (Read Uncommitted)
  • 回顾下读未提交概念

允许一个事务读取另一个事务尚未提交的数据。这种隔离级别提供了最高的并发性,但也可能导致一些问题,如脏读(Dirty Read)和不可重复读(Non-repeatable Read)

在这里插入图片描述

  • 读未提交事务1操作

事务1设置事务隔离级别为读未提交,开启事务,修改产品表id=1的price为99,注意这里并未提交事务

sql">-- 设置事务隔离级别为读未提交
SET GLOBAL TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
-- 开始事务
START TRANSACTION;
-- 更新表中id=1的产品价格
UPDATE product SET price=99 WHERE id=1;
-- 确认更新成功,读未提交事务1查询
SELECT * FROM product WHERE id=1;

在这里插入图片描述

  • 读未提交事务2操作

事务2设置事务隔离级别为读未提交,去查看产品表id=1的price,结果是99,也就是读到了事务1中还未提交事务的值

sql">-- 设置事务隔离级别为读未提交
SET GLOBAL TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
-- 开始事务
START TRANSACTION;
-- 读未提交事务2查询
SELECT * FROM product WHERE id=1;

在这里插入图片描述

2. 读已提交 (Read Committed)
  • 回顾下读已提交概念

读已提交它确保每个事务只能看到已经提交的更改,从而避免了脏读,但可能会发生不可重复读和幻读

在这里插入图片描述

  • 读已提交事务1操作

事务1设置事务隔离级别为读已提交,开启事务,修改产品表id=2的price为99,注意这里并未提交事务

sql">-- 设置事务隔离级别为读已提交
SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;
START TRANSACTION;
-- 更新表中id=2的数据
UPDATE product SET price=99 WHERE id=2;
-- 确认上面的UPDATE语句执行成功
SELECT * FROM product WHERE id=2;

在这里插入图片描述

  • 读已提交事务2操作

事务2设置事务隔离级别为读已提交,**去查看产品表id=2的price,结果是10,也就事务1更新前的值

sql">-- 设置事务隔离级别为读已提交
SET GLOBAL TRANSACTION ISOLATION LEVEL READ COMMITTED;
-- 开始事务
START TRANSACTION;
-- 读未提交事务2查询
SELECT * FROM product WHERE id=2;

在这里插入图片描述

  • 读已提交事务1再操作

将刚才上一步price更新为99的事务进行提交commit

sql">-- 提交事务
COMMIT;

在这里插入图片描述

  • 读已提交事务2再操作

事务1已提交,**去查看产品表id=2的price,结果是99,也就事务1更新后的值

sql">-- 读未提交事务2查询
SELECT * FROM product WHERE id=2;

在这里插入图片描述

3. 可重复读 (Repeatable Read)
  • 回顾下可重复读概念

可重复读确保在同一个事务内多次读取相同的数据时结果是一致的,即使在这期间有其他事务进行了更新。

在这里插入图片描述

  • 可重复读事务1操作

事务1设置事务隔离级别为可重复读,开启事务,修改产品表id=3的price为99,注意这里并未提交事务

sql">-- 设置事务隔离级别为可重复读
SET GLOBAL TRANSACTION ISOLATION LEVEL REPEATABLE READ;
START TRANSACTION;
-- 更新表中id=3的数据
UPDATE product SET price=99 WHERE id=3;
-- 确认上面的UPDATE语句执行成功
SELECT * FROM product WHERE id=3;

在这里插入图片描述

  • 可重复读事务2操作

事务2设置事务隔离级别为可重复读,**去查看产品表id=2的price,结果是10,也就事务1更新前的值

sql">-- 设置事务隔离级别为可重复读
SET GLOBAL TRANSACTION ISOLATION LEVEL REPEATABLE READ;
-- 开始事务
START TRANSACTION;
-- 读未提交事务3查询
SELECT * FROM product WHERE id=3;

在这里插入图片描述

  • 可重复读事务1再操作

将刚才上一步price更新为99的事务进行提交commit

sql">-- 提交事务
COMMIT;

在这里插入图片描述

  • 可重复读事务2再操作

事务1已提交,**去查看产品表id=3的price,结果还是10,也就事务1提交后也是重复读取提交前的10

sql">-- 可重复读事务3查询
SELECT * FROM product WHERE id=3;

在这里插入图片描述

4. 串行化 (Serializable)
  • 回顾下串行化概念

** 串行化确保事务以串行的方式执行**,完全避免了并发问题,但可能会降低系统的并发性能。

在这里插入图片描述

  • 串行化事务1操作

事务1设置事务隔离级别为串行化,开启事务,修改产品表id=1的price为99,注意这里并未提交事务

sql">-- 设置事务隔离级别为串行化
SET GLOBAL TRANSACTION ISOLATION LEVEL SERIALIZABLE;
-- 开始事务
START TRANSACTION;
-- 更新表中id=4的产品价格
UPDATE product SET price=99 WHERE id=4;
-- 确认更新成功,读未提交事务4查询
SELECT * FROM product WHERE id=1;

在这里插入图片描述

  • 串行化事务2操作

事务2设置事务隔离级别为串行化,去查看产品表id=4的price,结果一直阻塞中等待

sql">-- 设置事务隔离级别为读未提交
SET GLOBAL TRANSACTION ISOLATION LEVEL READ UNCOMMITTED;
-- 开始事务
START TRANSACTION;
-- 读未提交事务2查询
SELECT * FROM product WHERE id=1;

在这里插入图片描述

总结

本文通过生动的例子和实际操作,全面解读了MySQL中的四种事务隔离级别:读未提交、读已提交、可重复读和串行化

首先介绍了每种隔离级别的特点及其可能产生的问题,例如脏读、不可重复读和幻读。

接着详细说明了如何在MySQL中设置不同的隔离级别,包括全局设置和会话级别的设置方法

最后通过一系列实操案例,直观展示了不同隔离级别下事务间交互的具体表现,帮助读者理解各种隔离级别的实际应用场景及优缺点。

通过本文的学习,开发者可以更好地掌握事务隔离级别的选择与应用,从而提高数据库应用程序的一致性和性能

🍈猜你想问

如何与狗哥联系进行探讨
关注公众号【JavaDog程序狗】

公众号回复【入群】或者【加入】,便可成为【程序员学习交流摸鱼群】的一员,问题随便问,牛逼随便吹,目前群内已有超过300+个小伙伴啦!!!

2.踩踩狗哥博客

javadog.net

里面有狗哥的私密联系方式呦 😘

大家可以在里面留言,随意发挥,有问必答

在这里插入图片描述

🍯猜你喜欢

文章推荐

【实操】Spring Cloud Alibaba AI,阿里AI这不得玩一下(含前后端源码)

【规范】看看人家Git提交描述,那叫一个规矩

【项目实战】SpringBoot+uniapp+uview2打造H5+小程序+APP入门学习的聊天小项目

【项目实战】SpringBoot+uniapp+uview2打造一个企业黑红名单吐槽小程序

【模块分层】还不会SpringBoot项目模块分层?来这手把手教你!


http://www.ppmy.cn/news/1516594.html

相关文章

使用阿里的EasyExcel导入数据

工作遇到一种情况,在导入excel的时候数量过多,导致占用内存太大最终OOM.为了避免这样的情况再次出现,更换easyPoi为EasyExcel,它是一行一行读,非常节省内存且快速. 首先依赖 <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel-core<…

对想学习人工智能或者大模型技术从业者的建议

“ 技术的价值在于应用&#xff0c;理论与实践相结合才能事半功倍” 写这个关于AI技术的公众号也有差不多五个月的时间了&#xff0c;最近一段时间基本上都在保持日更状态&#xff0c;而且写的大部分都是关于大模型技术理论和技术方面的东西。‍‍‍‍‍‍‍‍‍ 然后最近一段…

Mozilla为本地音频到文本翻译开发Whisperfile引擎

Mozilla Ocho 小组正进行 Mozilla 的"创新和实验"。Llamafile 用于将大型语言模型以单个文件的形式发布&#xff0c;以便在不同的硬件/软件间轻松执行。Whisperfile 是一项将音频轻松转化为文本的新引擎。 正如其名称所暗示的&#xff0c;Whisperfile 是围绕 OpenAI…

RabbitMQ如果有100万消息堆积在MQ,如何解决(消息堆积如何解决)面试版

什么情况下产生消息堆积 消息堆积&#xff1a;当生产者发送消息的速度 超过了 消费者处理消息的速度&#xff0c;就会导致队列中的消息堆积。 消息堆积会产生的问题&#xff1a;直到队列存储的消息达到上限。之后发送的消息就会成为死信&#xff0c;可能会被丢弃。 解决消息…

数据库表的nb3和sql后缀的处理方式

后缀是sql的话就直接运行sql文件 就可以把数据库添加到本地了&#xff08;像这样&#xff09; 右键你选择存放的数据库 -- 运行sql文件 -- 选择后缀是sql的文件 如果同事给你了一个后缀是nb3的话 那么就需要你去还原了 你想把这个表加到哪个库下就选择 右键备份 -- 还原备份从…

数据库查询大量数据避免内存溢出的方法

原理就是分批查询。每次查询一定数量数据之后记录id&#xff0c;进行数据处理之后再继续查询继续处理&#xff0c; allFrameObject mapper.findAllFrameObjectByMaxId(minTime, beginRow, 1000); while (CollectionUtils.isNotEmpty(allFrameObject)) {beginRow allFrameObj…

如何构建小学至大学素质评价档案系统 —— php Vue 实践指南

&#x1f34a;作者&#xff1a;计算机毕设匠心工作室 &#x1f34a;简介&#xff1a;毕业后就一直专业从事计算机软件程序开发&#xff0c;至今也有8年工作经验。擅长Java、Python、微信小程序、安卓、大数据、PHP、.NET|C#、Golang等。 擅长&#xff1a;按照需求定制化开发项目…

GPIO(通用输入/输出)、中断(hal库)

目录 GPIO&#xff08;通用输入/输出)&#xff08;hal库&#xff09; GPIO工作模式 推挽输出&#xff08;Push-Pull Output&#xff09; 开漏输出&#xff08;Open-Drain Output&#xff09; 复用推挽输出&#xff08;Alternate Function Push-Pull Output&#xff09; 复…

金融涉案账户压降行动的实施成效与挑战

2024年上半年我国出台了关于金融行业相关管理办法 1 - 5 号令&#xff0c;不断完善相关法律法规&#xff0c;加强对欺诈行为的打击力度。加强了对互联网企业的监管力度&#xff0c;要求企业加强内部管理&#xff0c;建立健全用户信息保护机制&#xff0c;防止用户信息泄露和被滥…

Lora微调训练参数解读

前言 通过前面两次微调训练欺诈文本分类微调&#xff08;六&#xff09;&#xff1a;Lora单卡和欺诈文本分类微调&#xff08;七&#xff09;—— lora单卡二次调优&#xff0c;我们已经初步理解了微调的整个过程&#xff0c;里面涉及到不少的参数配置&#xff0c;这篇文章就对…

redis面试(二十三)写锁释放

先加了写锁&#xff0c;后面再次加写锁或者读锁 anyLock: { “mode”: “write”, “UUID_01:threadId_01:write”: 2, “UUID_01:threadId_01”: 1 } 写锁的释放lua脚本在这里 RedissonWriteLock.unlockInnerAsync() 比如说现在的参数是这 KEYS[1] anyLock KEYS[2] redi…

卖旧电脑前怎么彻底清除数据?卖旧电脑不留隐患

在科技日新月异的今天&#xff0c;电脑已成为我们日常生活和工作中不可或缺的工具。然而&#xff0c;随着技术的不断进步&#xff0c;我们可能会考虑更换新的电脑设备&#xff0c;而将旧的电脑出售或转让。 在卖旧电脑前&#xff0c;彻底清除电脑中的数据至关重要&#xff0c;…

leetcode46:全排列

全排列 给定一个不含重复数字的数组 nums &#xff0c;返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 List<List<Integer>> list new ArrayList<>();public List<List<Integer>> permute(int[] nums) {LinkedList<Integer> …

Shader 中的渲染顺序

1、深度测试和深度写入 有了深度测试和深度写入发挥作用让我们不需要关心不透明物体的渲染顺序比如一个物体A 挡住了 物体B&#xff0c;即使底层逻辑中 先渲染A&#xff0c;后渲染B&#xff0c;我们也不用担心 B的颜色会把A覆盖&#xff0c;因为在进行深度测试时&#xff0c;远…

Java-BatchProcessingUtil结合CompletableFuture工具类

为了结合批处理与 CompletableFuture 并使用自定义线程池,我们可以创建一个功能全面的工具类 BatchProcessingUtil。这个工具类将支持以下功能: 批处理遍历列表中的每个元素。 使用自定义线程池执行批处理任务。 返回一个 CompletableFuture 对象以支持异步回调和结果处理…

BlazeFace: Sub-millisecond Neural Face Detection on Mobile GPUs

Abstract 我们提出了BlazeFace&#xff0c;这是一种轻量级且性能优异的面部检测器&#xff0c;专为移动GPU推理而设计。它在旗舰设备上运行速度可达200到1000 FPS。这种超实时性能使其能够应用于任何增强现实管道中&#xff0c;作为任务特定模型的输入来准确识别面部感兴趣区域…

从数据生成到图数据库:Linux下Neo4j的CSV导入

文章目录 简介找到import文件夹准备csv表格数据导入neo4jTeacherStudent 简介 介绍如何在Linux系统中设置和使用Neo4j数据库。 首先&#xff0c;找到Neo4j的import文件夹&#xff0c;通常位于Neo4j安装目录下的data文件夹内&#xff0c;并展示通过neo4j.conf配置文件查找和修…

MySQL 数据库的规范化与反规范化详解

在数据库设计中&#xff0c;规范化&#xff08;Normalization&#xff09;和反规范化&#xff08;Denormalization&#xff09;是两个重要的概念&#xff0c;它们直接影响数据的存储效率、数据一致性以及查询性能。本文将详细介绍 MySQL 中的第一范式、第二范式和第三范式&…

qt圆环饼状图,非常小的窗口都能显示

非常小的窗口都能显示 QT core gui charts#include <QtCharts> using namespace QtCharts;//创建饼状图 void MainWindow::createpieSewies() {//饼状图QPieSeries * my_pieSeries new QPieSeries();//中间圆与大圆的比例my_pieSeries->setHoleSize(0.35);//…

Python知识点:如何使用PostgreSQL与Psycopg2进行数据库操作

要使用PostgreSQL与Psycopg2进行数据库操作&#xff0c;首先需要确保你的环境中已经安装了PostgreSQL和Psycopg2库。以下是一个简单的指南&#xff0c;展示如何使用Psycopg2与PostgreSQL进行连接和执行基本的数据库操作。 1. 安装Psycopg2 如果你还没有安装Psycopg2&#xff…