MySQL Innodb中 可重复读隔离级别是否能完全规避幻读

2024-04-14 15:12

本文主要是介绍MySQL Innodb中 可重复读隔离级别是否能完全规避幻读,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、MySQL 可重复读隔离级别下的幻读

MySQL Innodb引擎可重复读隔离级别下,已经尽可能最大程度的规避幻读的问题了,使得大多数情况下,重复读都是可以得到一致的结果。

针对于读数据,可以大致分为两种模式,快照读(select ... )保证每次数据读取相同,即使其他事务写入或更新了数据。另一种是当前读(select ... for update) 的方式,每次读取最新的数据。不过两种方式解决幻读的策略不同。

  • 快照读模式下,通过 MySQL MVCC 多版本机制解决,在执行第一个 select 语句后,会创建一个 Read View,后续的查询语句会先查询这个 Read View,通过 Read View 就可以在 undo log 版本链中找到事务开始的数据,所以整个过程中可以保证查询的数据都是一致的,即使中间被其他事务插入了新数据。

  • 当前读模式下,通过 MySQL Innodb 中的间隙锁和临键锁解决幻读,当执行 select ... for update 时,会针对相关的记录或间隙进行加锁,如果有其他事务在修改相关数据,或在间隙中新增数据,都会被阻塞,直到当前事务处理完成后才能继续操作,进而规避了幻读的发生。

不过上面两种方式虽然解决了大多数情况下的幻读问题,但是某些个别情况下还是不可避免会发生幻读。

例如有如下表结构:

CREATE TABLE `user` (`id` int NOT NULL AUTO_INCREMENT,`name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci DEFAULT NULL COMMENT '姓名',`age` int DEFAULT NULL COMMENT '年龄',`grade` int DEFAULT NULL COMMENT '等级',PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci;

写入两条测试数据:

INSERT INTO `testdb`.`user` (`id`, `name`, `age`, `grade`) VALUES (1, '张三', 18, 1);
INSERT INTO `testdb`.`user` (`id`, `name`, `age`, `grade`) VALUES (2, '李四', 18, 1);

二、可重复读隔离级别下幻读问题复现

首先在事务1中,查询 user 表下的所有数据:

BEGIN;
select * from `user`;

在这里插入图片描述
可以正常查到两条测试数据。

下面在事务2中插入一条新数据:

BEGIN;
INSERT INTO `user`(name,age,grade) value('王五', 18, 1);
COMMIT;

在这里插入图片描述

然后回到事务 1 ,当前如果继续执行查询,应该会命中先前查询的Read View

select * from `user`;

在这里插入图片描述
看到确实没有查出事务2新增的数据,但是下面在事务1中做一个更新操作,把所有 age > 15 的用户的 grade 修改成 2 ,然后再次执行上面相同的查询语句:

update `user` set grade = 2 where age > 15;
select * from `user`;

在这里插入图片描述
可以看到此时竟然查出了事务2新增的数据,对于事务1来说出现了幻读!

此时发生幻读主要在 update 执行后,首先之前的 select 属于快照读,不会对相关数据加锁,而 update 其实属于当前读,且 update 的范围,已经包含了事务2写入并提交的数据,所以导致后面的查询看到了新插入的数据。

同样的原理,将 update 语句换成 select .... for update 同样也会出现幻读。

三、如何解决幻读

从上面的例子可以看出,可重复读隔离级别并没有彻底解决幻读问题,只是尽可能最大程度的规避幻读的问题。其实有的时候我们可以从上层去协调这种问题,比如在代码层协调执行的过程。如果非要到SQL中解决,可以在最开始的 select 就加上 for update 或者 select ... in share mode,这样相关数据都被锁上了,其他事务的写操作就会被阻塞,缺点是处理的效率就会大大折扣,具体解决方法还是要结合业务决策。

这篇关于MySQL Innodb中 可重复读隔离级别是否能完全规避幻读的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/903313

相关文章

MySQL的JDBC编程详解

《MySQL的JDBC编程详解》:本文主要介绍MySQL的JDBC编程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、前置知识1. 引入依赖2. 认识 url二、JDBC 操作流程1. JDBC 的写操作2. JDBC 的读操作总结前言本文介绍了mysq

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

Python实现精确小数计算的完全指南

《Python实现精确小数计算的完全指南》在金融计算、科学实验和工程领域,浮点数精度问题一直是开发者面临的重大挑战,本文将深入解析Python精确小数计算技术体系,感兴趣的小伙伴可以了解一下... 目录引言:小数精度问题的核心挑战一、浮点数精度问题分析1.1 浮点数精度陷阱1.2 浮点数误差来源二、基础解决

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

从入门到精通详解Python虚拟环境完全指南

《从入门到精通详解Python虚拟环境完全指南》Python虚拟环境是一个独立的Python运行环境,它允许你为不同的项目创建隔离的Python环境,下面小编就来和大家详细介绍一下吧... 目录什么是python虚拟环境一、使用venv创建和管理虚拟环境1.1 创建虚拟环境1.2 激活虚拟环境1.3 验证虚

如何通过try-catch判断数据库唯一键字段是否重复

《如何通过try-catch判断数据库唯一键字段是否重复》在MyBatis+MySQL中,通过try-catch捕获唯一约束异常可避免重复数据查询,优点是减少数据库交互、提升并发安全,缺点是异常处理开... 目录1、原理2、怎么理解“异常走的是数据库错误路径,开销比普通逻辑分支稍高”?1. 普通逻辑分支 v

MySQL中On duplicate key update的实现示例

《MySQL中Onduplicatekeyupdate的实现示例》ONDUPLICATEKEYUPDATE是一种MySQL的语法,它在插入新数据时,如果遇到唯一键冲突,则会执行更新操作,而不是抛... 目录1/ ON DUPLICATE KEY UPDATE的简介2/ ON DUPLICATE KEY UP

MySQL分库分表的实践示例

《MySQL分库分表的实践示例》MySQL分库分表适用于数据量大或并发压力高的场景,核心技术包括水平/垂直分片和分库,需应对分布式事务、跨库查询等挑战,通过中间件和解决方案实现,最佳实践为合理策略、备... 目录一、分库分表的触发条件1.1 数据量阈值1.2 并发压力二、分库分表的核心技术模块2.1 水平分