设计素材网站知乎/百度seo排名优化公司推荐
1. 什么是幻读?
幻读是一种数据库事务中可能出现的并发问题,具体表现为:在同一个事务中,前后两次查询的结果集不同,仿佛“幻影”一般,出现了原本不存在的数据。
1.1 具体表现:
-
现象描述
事务 A 在某个范围内查询了多行记录,并对这些记录进行了修改。
随后,事务 B 在这个范围内插入了一条新记录。
当事务 A 再次查询时,会发现多了一行之前不存在的记录。 -
幻读 VS 脏读/不可重复读
- 脏读:事务读取到了另一个未提交事务的修改内容。
- 不可重复读:事务前后读取同一行数据时,内容发生了变化。
- 幻读:事务范围查询时,前后读取的结果集中多了新的“行”。
1.2 案例示例:
假设有一个 students
表,存储学生成绩记录:
id | name | score |
---|---|---|
1 | Alice | 85 |
2 | Bob | 90 |
-
事务 A:启动事务并执行以下查询:
SELECT * FROM students WHERE score > 80;
返回结果为:
id name score 1 Alice 85 2 Bob 90 -
事务 B:插入一条新记录:
INSERT INTO students (id, name, score) VALUES (3, 'Charlie', 88);
-
事务 A:再次执行相同查询:
SELECT * FROM students WHERE score > 80;
返回结果为:
id name score 1 Alice 85 2 Bob 90 3 Charlie 88
这时,事务 A 发现了一条“幻影”数据(Charlie
),这就是幻读。
2. 快照读如何避免幻读?
2.1 什么是快照读?
快照读指的是通过 MVCC(多版本并发控制)机制读取数据的一种方式,它读取的是某个时刻的“历史快照”,而不是当前最新的实际数据。
- 快照读依赖于 InnoDB 引擎实现的 MVCC。
- 使用快照读时,即使其他事务对数据进行了插入或修改,当前事务依然能看到事务开始时的数据状态。
2.2 快照读的特点
-
读取旧版本数据:
每次查询的结果是事务开始时生成的快照,不受其他事务影响。 -
无需加锁:
快照读不需要像“当前读”那样使用锁机制,效率更高。 -
典型语句:
对于常见的SELECT
语句,如果未显式使用锁机制,则是快照读。例如:SELECT * FROM students WHERE score > 80;
2.3 快照读如何避免幻读?
在 可重复读(REPEATABLE READ)隔离级别下,MySQL 使用 MVCC 为事务生成一致性视图。
- 每个事务在开始时,会记录当前的数据版本快照。
- 对于后续的
SELECT
操作,返回的结果总是基于这一快照,即使其他事务对数据进行了增删改,也不会影响快照读的结果。
举例说明:
假设在事务 A 开始时,students
表的状态如下:
id | name | score |
---|---|---|
1 | Alice | 85 |
2 | Bob | 90 |
-
事务 A:
START TRANSACTION; SELECT * FROM students WHERE score > 80;
返回结果为:
id name score 1 Alice 85 2 Bob 90 -
事务 B:插入一条新数据:
INSERT INTO students (id, name, score) VALUES (3, 'Charlie', 88);
-
事务 A:再次查询时:
SELECT * FROM students WHERE score > 80;
返回结果依然是事务开始时的快照:
id name score 1 Alice 85 2 Bob 90
这样,事务 A 避免了幻读,因为它读取的快照与事务开始时一致。
2.4 快照读的局限性
快照读仅适用于普通查询操作。如果事务需要对数据进行修改(如 UPDATE
、DELETE
、INSERT
),必须使用“当前读”,而当前读中可能会涉及幻读。
3. 当前读如何避免幻读?
3.1 什么是当前读?
当前读指的是读取数据库中最新的、实时的数据版本,通常发生在需要对数据进行修改的场景中。例如,UPDATE
、DELETE
、INSERT
等操作就会触发当前读。
- 当前读会通过加锁机制,确保数据的一致性。
典型语句:
SELECT * FROM students WHERE score > 80 FOR UPDATE;
3.2 当前读如何避免幻读?
在 可重复读(REPEATABLE READ)隔离级别下,MySQL 的当前读会引入锁机制(尤其是间隙锁),避免其他事务在同一范围内插入新数据,从而解决幻读问题。
3.3 间隙锁(Gap Lock)的作用
-
定义:
间隙锁是 InnoDB 在事务中为避免幻读问题而引入的一种锁,它锁定的是一个范围(gap),而不仅仅是具体的行。 -
作用:
- 防止其他事务在锁定范围内插入新记录。
- 确保同一事务中的前后查询结果集一致。
-
锁定范围示例:
假设查询条件为WHERE score > 80
,当前数据表如下:id name score 1 Alice 85 2 Bob 90 当事务 A 使用当前读时:
SELECT * FROM students WHERE score > 80 FOR UPDATE;
InnoDB 会锁定范围
(80, ∞)
,此时:- 其他事务不能在这个范围内插入新记录,例如
INSERT INTO students (id, name, score) VALUES (3, 'Charlie', 88)
会被阻塞。 - 保证当前事务不会读取到“幻影”数据。
- 其他事务不能在这个范围内插入新记录,例如
案例说明:
假设有如下数据:
id | name | score |
---|---|---|
1 | Alice | 85 |
2 | Bob | 90 |
-
事务 A:开启事务,并进行当前读:
START TRANSACTION; SELECT * FROM students WHERE score > 80 FOR UPDATE;
返回结果:
id name score 1 Alice 85 2 Bob 90 -
事务 B:试图插入一条数据:
INSERT INTO students (id, name, score) VALUES (3, 'Charlie', 88);
结果:事务 B 被阻塞,直到事务 A 提交或回滚。
-
事务 A:再次查询时:
SELECT * FROM students WHERE score > 80 FOR UPDATE;
返回的结果仍然是:
id name score 1 Alice 85 2 Bob 90
间隙锁保证了事务 A 的一致性,避免了幻读问题。
间隙锁的局限性:
-
可能引发性能问题:
- 间隙锁会阻止范围内的插入操作,如果锁定范围较大,可能造成事务等待时间过长。
-
事务隔离级别:
间隙锁只在 可重复读 隔离级别下有效,在更低的隔离级别(如读已提交)中不适用。
4. 幻读是否被完全解决?
尽管 可重复读(REPEATABLE READ)隔离级别通过快照读和当前读(间隙锁)在很大程度上避免了幻读,但并不能完全解决所有场景下的幻读问题。以下是详细的分析和示例。
4.1 幻读可能依然存在的场景
在一些复杂事务操作中,即使使用间隙锁,幻读问题依然可能发生。例如:
- 并发事务修改数据的交叉操作:涉及范围查询时,锁范围与实际查询范围不完全匹配。
- 间隙锁未覆盖所有查询条件:事务锁定的范围不能完全避免其他事务插入新的数据。
案例分析:复杂事务下的幻读
假设 students
表当前的状态为:
id | name | score |
---|---|---|
1 | Alice | 85 |
2 | Bob | 90 |
场景描述:
- 事务 A:希望对分数大于 80 的学生记录进行处理。
- 事务 B:在事务 A 操作期间插入了一条符合条件的记录。
具体操作:
-
事务 A:
START TRANSACTION; SELECT * FROM students WHERE score > 80 FOR UPDATE;
返回结果:
id name score 1 Alice 85 2 Bob 90 此时,间隙锁锁定了
(80, ∞)
范围。 -
事务 B:
在锁定范围之外插入了一条分数为 80 的记录:INSERT INTO students (id, name, score) VALUES (3, 'Charlie', 80);
结果:事务 B 的插入成功,因为分数为 80 不在事务 A 锁定的范围
(80, ∞)
内。 -
事务 A:
对分数大于等于 80 的记录进行处理:UPDATE students SET score = score + 5 WHERE score >= 80;
结果:事务 A 的操作覆盖了事务 B 插入的记录(
Charlie
),导致了一种幻读现象。
4.2 为什么幻读未被完全解决?
-
锁范围与查询条件不完全匹配:
间隙锁锁定的范围是基于查询条件计算的,但并不总是能够覆盖所有潜在的记录。 -
复杂的业务场景:
当查询涉及范围变化(如>=
、BETWEEN
),锁的逻辑变得复杂且难以避免边界情况。
4.3 如何完全解决幻读?
在某些关键场景下,为了避免幻读问题,可以使用更高的隔离级别或额外的手段:
-
可串行化隔离级别(SERIALIZABLE):
- 串行化隔离级别会对所有的读操作加锁,避免了所有类型的幻读问题。
- 代价:显著的性能开销,因为所有事务会变成串行执行。
-
手动加锁:
在事务操作中显式加表锁或行锁,确保范围锁定完全覆盖。 -
业务逻辑约束:
优化业务设计,避免范围查询可能带来的并发插入问题。
具体示例:可串行化避免幻读
事务 A:
SET TRANSACTION ISOLATION LEVEL SERIALIZABLE;
START TRANSACTION; SELECT * FROM students WHERE score >= 80 FOR UPDATE;
事务 B:
试图插入符合条件的记录:
INSERT INTO students (id, name, score) VALUES (3, 'Charlie', 88);
结果:事务 B 被阻塞,直到事务 A 提交或回滚,避免了幻读现象。
5. 总结
5.1 幻读问题的核心
幻读是事务中一种特殊的数据不一致现象,通常出现在并发环境下。当事务在范围查询中遇到其他事务插入或修改的符合条件的新数据时,可能会导致前后查询结果不一致,这种现象称为幻读。
5.2 如何避免幻读?
-
快照读:
- 使用多版本并发控制(MVCC),通过历史版本实现一致性读,从根本上避免了幻读。
- 快照读适用于查询操作,如
SELECT
。
-
当前读:
- 通过加锁(如间隙锁)避免并发插入或修改操作导致幻读问题。
- 当前读适用于更新操作,如
UPDATE
、DELETE
和带FOR UPDATE
的SELECT
。
-
间隙锁(Gap Lock):
- 锁定范围而非具体记录,阻止其他事务在锁定范围内插入新数据,从而在可重复读(REPEATABLE READ)隔离级别下避免幻读。
-
可串行化隔离级别(SERIALIZABLE):
- 通过更严格的锁机制,彻底避免了所有类型的幻读问题,但以性能开销为代价。
5.3 幻读是否被完全解决?
- 在可重复读隔离级别下,虽然快照读和间隙锁大大减少了幻读的发生,但仍然存在无法完全覆盖的场景,尤其是复杂的范围查询。
- 要彻底解决幻读问题,需要更高的隔离级别(如可串行化)或优化业务逻辑。
具体实例说明
-
在可重复读隔离级别下:
- 快照读避免了幻读,但不能适用于需要修改数据的场景。
- 当前读通过间隙锁避免了大多数幻读,但可能因锁范围不匹配而产生边界问题。
-
在可串行化隔离级别下:
- 完全阻止了幻读,但事务效率可能显著降低。
5.4 应对幻读的策略
-
选择合适的隔离级别:
- 大多数情况下,可重复读隔离级别(REPEATABLE READ)已经能够满足需求。
- 在需要强一致性时,考虑使用可串行化隔离级别。
-
优化锁机制:
- 根据具体业务需求,合理选择加锁范围,避免过大或过小。
-
业务逻辑调整:
- 通过限制并发操作或使用显式锁,降低幻读发生的概率。
相关文章:

MySQL中的幻读问题
1. 什么是幻读? 幻读是一种数据库事务中可能出现的并发问题,具体表现为:在同一个事务中,前后两次查询的结果集不同,仿佛“幻影”一般,出现了原本不存在的数据。 1.1 具体表现: 现象描述 事务 A…...

AI后端工程师面试题的内容
AI后端工程师面试题主要包括以下几个方面的内容: 一、技术基础和项目经验: 1. 微服务架构的理解和应用:请描述你对微服务架构的理解,并举例说明一个你参与过的微服务项目,阐述你在该项目中扮演的角色和所承…...

MFC工控项目实例三十五读取数据库数据
点击按钮打开文件夹中的数据文件生成曲线 相关代码 void CSEAL_PRESSUREDlg::OnTesReport() {CFileDialog dlgOpen(TRUE/*TRUE打开,FALSE保存*/,0,0,OFN_NOCHANGEDIR|OFN_FILEMUSTEXIST,"All Files(mdb.*)|*.*||",//文件过滤器NULL);CString mdb_1, m…...

OpenWrt -制作ubifs文件系统的固件
目的 创建一个ubifs为文件系统的镜像 将backup目录中的内容打包成ubifs文件系统。 ubifs的分区定义 ubi-backup.cfg 文件内容如下, [backup] modeubi imagenand-ipq6018-single.img vol_id0 vol_typedynamic vol_namebackup [bkver] modeubi imagebackup.ubifs v…...

C++ - 继承
继承的基本概念 继承就是一种代码的复用. 子类通过继承父类, 就能使用父类的变量, 方法. 学生和老师这两种身份, 他们都有共同的属性: 他们都有名称, 年龄, 性别 .... 当然他们也有各种独有的属性, 学生有学号, 老师有工号 .... 对于这些共有的属性, 我们可以将它们提取出来: …...

华为服务器使用U盘重装系统
一、准备工作 下载官方系统(注意服务器CPU的架构是x86-64还是aarch64,不然可能报意想不到的错)制作启动U盘(下载rufus制作工具,注意文件系统选FAT32还是NTFS) 二、安装步骤 将U盘插入USB接口重启服务器…...

网络分层模型( OSI、TCP/IP、五层协议)
1、网络分层模型 计算机网络是一个极其复杂的系统。想象一下最简单的情况:两台连接在网络上的计算机需要相互传输文件。不仅需要确保存在一条传输数据的通路,还需要完成以下几项工作: 发起通信的计算机必须激活数据通路,这包括发…...

前端开发 之 15个页面加载特效上【附完整源码】
文章目录 一:彩球环绕加载特效1.效果展示2.HTML完整代码 二:跷跷板加载特效1.效果展示2.HTML完整代码 三:两个圆形加载特效1.效果展示2.HTML完整代码 四:半环加载特效1.效果展示2.HTML完整代码 五:音乐波动加载特效1.效…...

Spring Boot使用JDK 21虚拟线程
JDK 21引入的虚拟线程(Virtual Threads)是 Project Loom 的一部分,旨在显著简化并发编程并提高 Java 应用的可扩展性。以下是虚拟线程的主要特点: 1. 概念 虚拟线程是轻量级线程,与传统的操作系统线程不同࿰…...

《从0到1常用Map集合核心摘要 + 不深不浅底层核心》
《从0到1常用Map集合核心摘要不深不浅底层核心》 前置知识 什么是键值对 键值对是一种数据结构,键是唯一标识,值是对应数据,用来快速查找信息。例: {"name": "Alice"},键是name,…...

12 设计模式之工厂方法模式
一、什么是工厂方法模式? 1.定义 在软件开发中,设计模式 是解决常见软件设计问题的最佳实践。而 工厂方法模式(Factory Method Pattern) 作为创建型设计模式之一,常常被用来解决对象创建问题。它通过将对象的创建交给…...

spaCy 入门与实战:强大的自然语言处理库
spaCy 入门与实战:强大的自然语言处理库 spaCy 是一个现代化、工业级的自然语言处理(NLP)库,以高效、易用和功能丰富著称。它被广泛应用于文本处理、信息提取和机器学习任务中。本文将介绍 spaCy 的核心功能,并通过一…...

python包的管理和安装——笔记
1.列出包 pip list pip freeze 用这2个可以查看当前python 下所有的包和版本,还有下载地址 如果只是想导出当前的环境 可以用 2.安装pipreqs pip install pipreqs,pipreqs ./可以导出当前项目的包这个包 遇到编码报错 pipreqs ./ --encodingutf8 p…...

Vue前端页面内嵌套本项目iframe窗口的通信传输方式
一、目的 想要在iframe中使用本项目页面、并能够与其父页面组件实现实时通信。Vue前端页面内嵌套本项目iframe窗口的通信传输方式-星林社区 https://www.jl1mall.com/forum/PostDetail?postId20241202172800023969 二、iframe通信方式 1.接收消息 页面需要监听 message 事件…...

【WEB开发.js】addEventListener事件监听器的绑定和执行次数的问题(小心踩坑)
假设我们有一个按钮,用户点击该按钮后,会选择一个文件,且我们希望每次点击按钮时只触发一次文件处理。下面我会给你一个简单的例子,展示放在函数内部和放在函数外部的区别。 1. 将事件监听器放在函数内部(问题的根源&…...

用于LiDAR测量的1.58um单芯片MOPA(一)
--翻译自M. Faugeron、M. Krakowski1等人2014年的文章 1.简介 如今,人们对高功率半导体器件的兴趣日益浓厚,这些器件主要用于遥测、激光雷达系统或自由空间通信等应用。与固态激光器相比,半导体器件更紧凑且功耗更低,这在低功率供…...

【GPT】代谢概念解读
以下是对代谢中分解代谢和合成代谢两个概念的深入解读,用简单易懂的方式展开说明: 1. 分解代谢(Catabolism) 什么是分解代谢? 分解代谢是身体把大分子“拆开”的过程。就像把一个三明治分解成面包片、肉片和菜叶&#…...

Devops-git篇-01-git环境配置
环境配置 设置用户签名 配置用户名: git config --global user.name 你的用户名 配置邮箱: git config --global user.email 注册的邮箱 配置好之后,可以用git config --global --list命令查看配置是否OK $ git config --global --list u…...

STM32 HAL库开发学习1.STM32CubeMX 新建工程
STM32 HAL库开发学习1.STM32CubeMX 新建工程 一、 STM32 CubeMX 下载二、CubeMX 功能介绍1. 固件包路径设置2. 新建工程 三、创建项目实例1. 新建项目2. GPIO 管脚设置3. GPIO 窗口配置4. 调试设置5. 时钟配置6. 项目管理(1)项目信息(2&#…...

JS学习(2)(浏览器执行JS过程、JS的ECMAScript、DOM、BOM)
目录 一、浏览器如何执行JS? (1)浏览器主要的组成部分。 1、渲染引擎。 2、JS引擎。 (2)演示。 二、JS的组成。 (1)JS主要由三部分组成。 1、JS基础。 2、JS-API。 (2)EC…...

如何解决服务器扫描出的ASP木马问题
随着互联网的发展,网站安全问题日益凸显。其中,ASP(Active Server Pages)木马因其隐蔽性和危害性成为攻击者常用的手段之一。本文将详细介绍如何检测和清除服务器上的ASP木马,以保障网站的安全。 1. ASP木马概述 ASP…...

SpringBoot 架构助力夕阳红公寓管理系统可持续发展战略
摘 要 如今社会上各行各业,都在用属于自己专用的软件来进行工作,互联网发展到这个时候,人们已经发现离不开了互联网。互联网的发展,离不开一些新的技术,而新技术的产生往往是为了解决现有问题而产生的。针对于夕阳红公…...

TCP、HTTP、RPC
一、TCP (Transmission Control Protocol) 定义 TCP(传输控制协议)是一种面向连接、可靠传输的传输层协议,用于在计算机网络中提供端到端的数据通信服务。它是互联网协议套件的一部分,与IP(互联网协议)一…...

《C++ 中 RNN 及其变体梯度问题的深度剖析与解决之道》
在当今人工智能蓬勃发展的浪潮中,递归神经网络(RNN)及其变体长短期记忆网络(LSTM)和门控循环单元(GRU)在处理序列数据方面展现出了强大的潜力。然而,当我们在 C中着手实现这些网络时…...

TypeScript 在 React 中的应用
文章目录 前言一、为什么要在 React 中使用 TypeScript?二、如何在React中使用 TypeScript三、高级类型结语 前言 随着前端开发的复杂度不断提升,开发者对于代码质量、可维护性和开发效率的要求也日益增高。TypeScript 作为一种为 JavaScript 添加静态类…...

黑马2024AI+JavaWeb开发入门Day07-部门管理-日志技术飞书作业
视频地址:哔哩哔哩 讲义作业飞书地址:day07作业 完成新增班级和查询班级的接口开发 1、ClazzController.java package org.example.controller;import lombok.extern.slf4j.Slf4j; import org.example.pojo.Clazz; import org.example.service.Clazz…...

UIlicious - 自动化端到端测试
在现代软件开发中,测试自动化已然成为产品交付质量的基石。而端到端测试(E2E),作为验证整个应用流畅运行的关键,常常是测试工作中最具挑战性的一环。这时,一款简单高效的自动化测试工具——UIlicious&#…...

JMeter中获取随机数、唯一ID、时间日期(包括当前日期增减)截取指定位数的字符等
在JMeter中,您可以使用内置的函数和一些额外的插件来获取随机数、唯一ID、时间日期以及截取指定位数的字符。以下是一些常用的方法: 获取随机数: 使用__Random函数,您可以在指定的最小值和最大值之间生成一个随机数。例如…...

构建自己的docker的ftp镜像
aarch64系统可运行的docker镜像 构建自己的vsftpd镜像,我是在windows系统下的docker desktop中构建运行于aarch64 GNU/Linux系统的ftp镜像。 系统环境: Welcome to Debian GNU/Linux with Linux x.x.x dockerfile FROM ubuntu:latestUSER rootRUN ap…...

人机交互革命,为智能座舱市场激战注入一针「催化剂」
从AIGC到AGI赋能,智能座舱人机交互体验迎来新范式。 不断训练、迭代的大模型,为智能座舱带来了更全面的感知能力、更准确的认知理解,以及更丰富的交互模态,显著提升了其智能化水平。 “AI大模型的快速应用与迭代,推动…...