【每日八股】复习 MySQL Day1:事务
文章目录
- 复习 MySQL Day1:事务
- MySQL 事务的四大特性?
- 并发事务会出现什么问题?
- MySQL 事务的隔离级别?
- 不同事务隔离级别下会发生什么问题?
- MVCC 的实现原理?
- 核心数据结构
- 版本链构建示例
- 可见性判断算法
- MVCC 可见性判断总结
- 幻读如何解决?
- 读已提交隔离级别如何实现?
复习 MySQL Day1:事务

MySQL 事务的四大特性?
- 原子性:事务当中的若干条数据库操作要么全部成功,要么全部失败;
- 一致性:数据库总是从一个一致性的状态迁移到另一个一致性的状态;
- 隔离性:事务提交之前,对数据库做出的更改对其他事务不可见;
- 持久性:事务提交之后就会被持久化到磁盘当中。
并发事务会出现什么问题?
指的主要是由 MySQL 隔离级别的不同带来的若干问题。
- 脏读:事务还未提交,其所做的修改就已经可以被其他事务所读取了;
- 不可重复读:事务开始时读取的数据与事务执行过程中读取到的数据不同;
- 幻读:通常发生在区间查询的场景下,指的是在一个事务执行期间内,上一次范围查询读取到的数据与本次范围查询读取到的数据不一致,其发生的原因可能是在事务执行期间有其他事务向第一次查询的数据区间当中插入了新的数据。
MySQL 事务的隔离级别?
- 读未提交:事务还未提交,其他事务就可看到其所做的更改,隔离级别最低;
- 读已提交:事务提交之后,其所做的更改才能够被其他事务所看到。在读已提交隔离级别下,当前事务可以看到其他事务执行完成后所做的修改;
- 可重复读:MySQL InnoDB 数据引擎的默认隔离级别,指的是从事务开始到事务结束期间,读取到的数据都是一致的;
- 串行化:对记录加上读写锁,在多个事务进行读写的过程中,如果发生了锁冲突,那么当前事务必须等上一个事务读写完成方可进行读写。串行化不存在并发事务的问题,但它的并发性能最差。
不同事务隔离级别下会发生什么问题?
- 读未提交:脏读、不可重复读、幻读;
- 读已提交:不可重复读、幻读;
- 可重复读:幻读;
- 串行化:不存在并发事务的问题;
MVCC 的实现原理?
MVCC 是数据库实现并发控制的关键技术,InnoDB 数据引擎通过 MVCC 实现读操作的并发,极大提高了数据库的并发性能。
核心数据结构
隐藏字段
InnoDB 为每行记录添加了以下隐藏字段:
- DB_TRX_ID:记录创建或最后一次修改该行记录的事务 ID;
- DB_ROLL_PTR:回滚指针,指向 undo log 记录;
- DB_ROW_ID:隐含的自增行 ID;
- DELETE BIT:记录该行是否删除。
Undo Log(回滚日志)
- 存储行记录的历史版本;
- 组成版本链,通过 DB_ROLL_PTR 指针连接;
- 用于事务回滚与 MVCC 读取。
Read View(读视图)
- m_ids:生成 Read View 时活跃的事务 ID 列表;
- min_trx_id:m_ids 中的最小事务 ID;
- max_trx_id:m_ids 中的最大事务 ID;
- create_trx_id:创建该 Read View 的事务 ID。
版本链构建示例
- 每次更新操作都会在 undo log 记录旧数据版本;
- 通过 DB_ROLL_PTR 指针形成单向链表;
- 链表头是最新版本,尾部是最旧版本。
可见性判断算法
比较 DB_TRX_ID 与 creator_trx_id
如果相等,说明当前记录是该事务自身修改的事务,对当前事务可见。
检查 DB_TRX_ID < min_trx_id
说明这条记录在当前 Read View 生成之前已经提交,对当前事务可见。
检查 DB_TRX_ID >= max_trx_id
说明这条记录是在 Read View 生成之后创建的,对当前事务不可见。
检查 DB_TRX_ID 是否在 m_ids 事务活跃列表当中
- 存在:生成 Read View 时当前记录仍活跃,对当前事务不可见;
- 不存在:最后一次修改该条记录的事务已提交,对当前事务可见;
MVCC 可见性判断总结
总的来说,针对基于 MVCC 的事务可见性判断,关键的字段包括以下几个:
- 对于记录,每一条记录都有一个隐式的 DB_TRX_ID 字段,用于记录最后一个修改这条记录的事务 ID;
- 对于 SELECT 操作生成的 Read View,其隐式包含以下几个字段:
1)m_ids:Read View 生成时活跃的事务 ID 列表;
2)min_trx_id:m_ids 中最小的事务 ID;
3)max_trx_id:m_ids 中最大的事务 ID;
4)create_trx_id:创建这个 Read View 的事务 ID。
可见性判断的算法流程如下:
- 首先对比记录的 DB_TRX_ID 和 creator_trx_id,相等则代表该记录最后一次由当前视图修改,对该事务可见;
- 之后再比对 DB_TRX_ID 和 min_trx_id 以及 max_trx_id 的大小,如果小于 min_trx_id,说明修改该记录的事务在生成 Read View 时已提交,对当前事务可见;如果大于 max_trx_id,说明修改该记录的事务在当前事务之后创建,其所做的修改对当前事务不可见,通过 DB_ROLL_PTR 找到该记录的上一个版本。
- 最后查看 DB_TRX_ID 是否在 m_ids 当中,如果在,说明修改这条记录的事务在活跃列表当中,该记录的当前版本对当前事务不可见;否则说明修改该记录的事务已经提交,这条记录对当前事务可见。
幻读如何解决?
通过快照读(一致性非锁定读)
对于普通的 SELECT 查询语句,InnoDB 使用 MVCC 提供一致性视图,避免看到其他事务插入的数据。
使用间隙锁(Gap Lock)和临键锁(Next-Key Lock)
InnoDB 在可重复读隔离级别下通过以下方式防止幻读:
- 间隙锁(Gap Lock):锁定索引记录之间的间隙;
- 临键锁(Next-Key Lock):临键锁是记录锁(行锁)+ 间隙锁的组合,锁定记录及其前面的间隙。
一个基于间隙锁 + 临键锁防止幻读的例子如下:
-- 事务1
BEGIN;
SELECT * FROM users WHERE age > 20 FOR UPDATE; -- 锁定age>20的所有记录和间隙
-- ⬆️ 显式地使用区间锁锁定一个范围, 避免在范围内有新记录插入
-- 此时事务2尝试插入age>20的记录会被阻塞
INSERT INTO users(name, age) VALUES('new_user', 25); -- 阻塞
总结
使用「MVCC 版本控制」或「间隙锁 + 临键锁」这两种方式可以避免幻读的问题。
读已提交隔离级别如何实现?
在读已提交隔离级别下,每次执行 SELECT 语句都会创建一个 Read View。创建 Read View 时已经提交的事务所做的修改对当前事务是可见的(会导致不可重复读问题),但未提交以及当前事务之后的事务所做的修改不可见。
相关文章:
【每日八股】复习 MySQL Day1:事务
文章目录 复习 MySQL Day1:事务MySQL 事务的四大特性?并发事务会出现什么问题?MySQL 事务的隔离级别?不同事务隔离级别下会发生什么问题?MVCC 的实现原理?核心数据结构版本链构建示例可见性判断算法MVCC 可…...
java 设计模式之代理模式
简介 代理模式:使用代理类来增强目标类的功能。在代码结构上,代理对象持有目标对象,通过代理对象访问目标对象,这样可以在不改变目标对象的前提下增加额外的功能,如权限校验、缓存等 代理模式内部的角色:…...
外接键盘与笔记本命令键键位不同解决方案(MacOS)
文章目录 修改键位第一步:打开设置第二步:进入键盘快捷键第三步:修改修饰键设置第四步:调整键位第五步:保存设置tips ikbc c87键盘win键盘没反应的解决亲测的方法这是百度的答案标题常规组合键尝试:型号差…...
python爬虫复习
requests模块 爬虫的分类 通用爬虫:将一整张页面进行数据采集聚焦爬虫:可以将页面中局部或指定的数据进行采集 聚焦爬虫是需要建立在通用的基础上来实现 功能爬虫:基于selenium实现的浏览器自动化的操作分布式爬虫:使用分布式机群…...
kotlin知识体系(五) :Android 协程全解析,从作用域到异常处理的全面指南
1. 什么是协程 协程(Coroutine)是轻量级的线程,支持挂起和恢复,从而避免阻塞线程。 2. 协程的优势 协程通过结构化并发和简洁的语法,显著提升了异步编程的效率与代码质量。 2.1 资源占用低(一个线程可运行多个协程)…...
vscode stm32 variable uint32_t is not a type name 问题修复
问题 在使用vscodekeil开发stm32程序时,发现有时候vscode的自动补全功能失效,且problem窗口一直在报错。variable “uint32_t” is not a type name uint32_t 定义位置 uint32_t 实际是在D:/Keil_v5/ARM/ARMCC/include/stdint.h中定义的。将D:/Keil_v5…...
Formality:Bug记录
相关阅读 Formalityhttps://blog.csdn.net/weixin_45791458/category_12841971.html?spm1001.2014.3001.5482 本文记录博主在使用Synopsys的形式验证工具Formality中遇到的一个Bug。 Bug复现 情况一 // 例1 module dff (input clk, input d_in, output d_out …...
在ubuntu20.04+系统部署VUE及Django项目的过程记录——以腾讯云为例
目录 1. 需求2. 项目准备3. VUE CLI项目部署3.1 部署前的准备3.1.1 后端通信路由修改3.1.2 导航修改 3.2 构建项目3.3 配置nginx代理 4. 后端配置4.1 其他依赖项4.2 单次执行测试4.3 创建Systemd 服务文件4.4 配置 Nginx 作为反向代理 5. 其他注意事项 1. 需求 近期做一些简单…...
回归,git 分支开发操作命令
核心分支说明 主分支(master/production)存放随时可部署到生产环境的稳定代码,仅接受通过测试的合并请求。 开发分支(develop)集成所有功能开发的稳定版本,日常开发的基础分支,从该分支创建特性…...
【java+Mysql】学生信息管理系统
学生信息管理系统是一种用于管理学生信息的软件系统,旨在提高学校管理效率和服务质量。本课程设计报告旨在介绍设计和实现学生信息管理系统的过程。报告首先分析了系统的需求,包括学生基本信息管理、成绩管理等功能。接着介绍了系统的设计方案࿰…...
小白从0学习网站搭建的关键事项和避坑指南(2)
以下是针对小白从零学习网站搭建的 进阶注意事项和避坑指南(第二期),覆盖开发中的高阶技巧、常见陷阱及解决方案,帮助你在实战中提升效率和质量: 一、进阶技术选型避坑 1. 前端框架选择 误区:盲目追求最新…...
Windows 10 上安装 Spring Boot CLI详细步骤
在 Windows 10 上安装 Spring Boot CLI 可以通过以下几种方式完成。以下是详细的步骤说明: 1. 手动安装(推荐) 步骤 1:下载 Spring Boot CLI 访问 Spring Boot CLI 官方发布页面。下载最新版本的 .zip 文件(例如 sp…...
spring boot -- 配置文件application.properties 换成 application.yml
在Spring Boot项目中,application.properties和application.yml是两种常用的配置文件格式,它们各自具有不同的特点和适用场景2。以下是它们之间的主要差异2: 性能差异 4: 加载机制 2: application.properties文件会被加载到内存中,并且只加载一次,之后直接从内存中读取…...
Spring Boot实战:基于策略模式+代理模式手写幂等性注解组件
一、为什么需要幂等性? 核心定义:在分布式系统中,一个操作无论执行一次还是多次,最终结果都保持一致。 典型场景: 用户重复点击提交按钮网络抖动导致的请求重试消息队列的重复消费支付系统的回调通知 不处理幂等的风…...
【Rust 精进之路之第14篇-结构体 Struct】定义、实例化与方法:封装数据与行为
系列: Rust 精进之路:构建可靠、高效软件的底层逻辑 作者: 码觉客 发布日期: 2025-04-20 引言:超越元组,给数据赋予意义 在之前的学习中,我们了解了 Rust 的基本数据类型(标量)以及两种基础的复合类型:元组 (Tuple) 和数组 (Array)。元组允许我们将不同类型的值组合…...
postgres 数据库信息解读 与 sqlshell常用指令介绍
数据库信息: sqlshell Server [localhost]: 192.168.30.101 Database [postgres]: Port [5432]: 5432 Username [postgres]: 用户 postgres 的口令: psql (15.12, 服务器 16.8 (Debian 16.8-1.pgdg120+1)) 警告:psql 主版本15,服务器主版本为16.一些psql功能可能无法正常使…...
论文阅读:2024 arxiv DeepInception: Hypnotize Large Language Model to Be Jailbreaker
总目录 大模型安全相关研究:https://blog.csdn.net/WhiffeYF/article/details/142132328 DeepInception: Hypnotize Large Language Model to Be Jailbreaker DeepInception:催眠大型语言模型,助你成为越狱者 https://arxiv.org/pdf/2311.0…...
vue2技术练习-开发了一个宠物相关的前端静态商城网站-宠物商城网站
为了尽快学习掌握相关的前端技术,最近又实用 vue2做了一个宠物行业的前端静态网站商城。还是先给大家看一下相关的网站效果: 所以大家如果想快速的学习或者掌握一门编程语言,最好的方案就是通过学习了基础编程知识后,就开始利用…...
嵌入式学习——远程终端登录和桌面访问
目录 通过桥接模式连接虚拟机和Windows系统 1、桥接模式 2、虚拟机和Windows连接(1) 3、虚拟机和Windows连接(2) 在Linux虚拟机中创建新用户 Windows系统环境下对Linux系统虚拟机操作 远程登录虚拟机(1ÿ…...
wpf stylet框架 关于View与viewmodel自动关联绑定的问题
1.1 命名规则 Aview 对应 AVIewModel, 文件夹 views 和 viewmodels 1.2 需要注册服务 //RootViewModel是主窗口 public class Bootstrapper : Bootstrapper<RootViewModel>{/// <summary>/// 配置IoC容器。为数据共享创建服务/// </summary…...
如何新建一个空分支(不继承 master 或任何提交)
一、需求分析: 在 Git 中,我们通常通过 git branch 来新建分支,这些分支默认都会继承当前所在分支的提交记录。但有时候我们希望新建一个“完全干净”的分支 —— 没有任何提交,不继承 master 或任何已有内容,这该怎么…...
HarmonyOS-ArkUI-动画分类简介
本文的目的是,了解一下HarmonyOS动画体系中的分类。有个大致的了解即可。 动效与动画简介 动画,是客户端提升界面交互用户体验的一个重要的方式。可以使应用程序更加生动灵越,提高用户体验。 HarmonyOS对于界面的交互方面,围绕回归本源的设计理念,打造自然,流畅品质一提…...
Qt编写推流程序/支持webrtc265/从此不用再转码/打开新世界的大门
一、前言 在推流领域,尤其是监控行业,现在主流设备基本上都是265格式的视频流,想要在网页上直接显示监控流,之前的方案是,要么转成hls,要么魔改支持265格式的flv,要么265转成264,如…...
[第十六届蓝桥杯 JavaB 组] 真题 + 经验分享
A:逃离高塔(AC) 这题就是简单的签到题,按照题意枚举即可。需要注意的是不要忘记用long,用int的话会爆。 📖 代码示例: import java.io.*; import java.util.*; public class Main {public static PrintWriter pr ne…...
深⼊理解 JVM 执⾏引擎
深⼊理解 JVM 执⾏引擎 其中前端编译是在 JVM 虚拟机之外执⾏,所以与 JVM 虚拟机没有太⼤的关系。任何编程语⾔,只要能够编译出 满⾜ JVM 规范的 Class ⽂件,就可以提交到 JVM 虚拟机执⾏。⾄于编译的过程,如果你不是想要专⻔去研…...
iwebsec靶场 文件包含关卡通关笔记11-ssh日志文件包含
目录 日志包含 1.构造恶意ssh登录命令 2.配置ssh日志开启 (1)配置sshd (2)配置rsyslog (3)重启服务 3.写入webshell木马 4.获取php信息渗透 5.蚁剑连接 日志包含 1.构造恶意ssh登录命令 ssh服务…...
kafka菜鸟教程
一、kafka原理 1、kafka是一个高性能的消息队列系统,能够处理大规模的数据流,并提供低延迟的数据传输,它能够以每秒数十万条消息的速度进行读写操作。 二、kafka优点 1、服务解耦 (1)提高系统的可维护性 通过服务…...
应用镜像是什么?轻量应用服务器的镜像大全
应用镜像是轻量应用服务器专属的,镜像就是轻量应用服务器的装机盘,应用镜像在原有的纯净版操作系统上集成了应用程序,例如WordPress应用镜像、宝塔面板应用镜像、WooCommerce等应用,阿里云服务器网aliyunfuwuqi.com整理什么是轻量…...
深入理解分布式缓存 以及Redis 实现缓存更新通知方案
一、分布式缓存简介 1. 什么是分布式缓存 分布式缓存:指将应用系统和缓存组件进行分离的缓存机制,这样多个应用系统就可以共享一套缓存数据了,它的特点是共享缓存服务和可集群部署,为缓存系统提供了高可用的运行环境,…...
Spring Boot 中的自动配置原理
2025/4/6 向全栈工程师迈进! 一、自动配置 所谓的自动配置原理就是遵循约定大约配置的原则,在boot工程程序启动后,起步依赖中的一些bean对象会自动的注入到IOC容器中。 在讲解Spring Boot 中bean对象的管理的时候,我们注入bean对…...
