undo日志详解

一、undo日志介绍

上一节详细的说了redo日志,redo日志的功能就是把增删改操作都记录着,如果断电导致内存中的脏页丢失,可以根据磁盘中的redo日志文件进行恢复。redo日志被设计出来是为了保证数据库的持久性,undo日志设计出来是为了保证数据库的原子性。一个事务开启之后,做增删改操作,会记录redo日志,但是如果后悔了进行回滚,redo日志是做不到回退的,这时候就需要undo日志了。

总结:记住一句话,redo日志保证了数据库的持久性,undo日志保证了数据库的原子性。

二、undo日志格式

一般每次对记录做一次改动就会对应生成一条undo日志。有些更新语句记录的操作中,也可能会生成两条undo日志。

一个事务在执行过程中可能新增,删除,更新若干条记录,同样也会生成若干条记录,这些undo日志会从0开始编号,并且是按照顺序生成,被称为第0号undo日志,1号。。。

这些undo日志是被记录到类型为FIL_PAGE_UNDO_LOG页面。这些页面可以从系统表空间(ibdata1文件)中分配,也可以从一种专门存放undo日志的表空间,也就是undo tablespace。

insert操作对应的undo日志

前面文章说过,插入操作分为乐观插入和悲观插入。但是关于插入操作无非就是把记录放到数据页中,如果想要回滚就直接把这条记录删除就可以,也就是说在写对应的undo日志时,主要是把这条记录的主键记录,这样删除记录就可以根据主键找到二级索引和聚簇索引上的记录。InnoDB设计了一个类型为TRX_UNDO_INSERT_RECundo⽇志

属性说明:

undo no上面说过,从0开始,只要这个事务没提交,按顺序生成undo日志,依次递增。

主键的每个列占用的存储空间和真实值。为什么存储主键就可以了,上面也说了,如果想要回滚就直接把这条记录删除就可以,也就是说在写对应的undo日志时,主要是把这条记录的主键记录,这样删除记录就可以根据主键找到二级索引和聚簇索引上的记录。后面的delete和update也一样不再强调。

undo_demo 中插⼊两条记录:
BEGIN;  # 显式开启⼀个事务,假设该事务的 id 100
# 插⼊两条记录
INSERT INTO undo_demo(id, key1, col)
    VALUES (1, 'AWM' , ' 狙击枪 ' ), (2, 'M416' , ' 步枪 ' );

因为主键只包含id,所以插入的两条记录,会按照顺序生成两个undo日志,第一个undo日志:

第二个undo日志:

roll_pointer隐藏列的含义

前面的文章介绍过记录的三个隐藏列,只剩roll_pointer没有说过,他本质上就是一个指向记录对应的undo日志的一个指针,比方说上面的插入两条记录的操作,每条记录都有对应的一个undo日志。关系就是记录被存储到类型为FIL_PAGE_INDEX的页面(前面说的数据页),undo日志被放到类型为FIL_PAGE_UNDO_LOG的页面中

delete操作对应的undo日志

前面的文章详细的说过数据页中的记录会组成一个单向链表,我们把这个链表称之为正常记录链表。并且删除某条记录,被删除的记录也会形成一个链表,叫垃圾链表,并且这条垃圾链表中记录占用的存储空间可以被重新利用。

上图简化了记录,只标记了delete_mask属性,我们删除最后一条记录,删除的过程需要经历两个阶段:

阶段一:只是将该记录的delete_mask表示为设置为1,修改记录隐藏列trx_id、roll_pointer这些隐藏列的值。

在事务提交之前,删除的记录都处于这个中间状态,被删除的记录没有被加入垃圾链表(为了实现MVCC功能)。

阶段二:当删除的语句所在事务提交之后,会有专门的线程把记录加入垃圾链表中,并且调整页面的其他信息。这个阶段称为purge。

把阶段二执行完了,这条记录就算真正的删除了,这条已删除的记录占用的存储空间也可以被重新利用了。

从上图我们可以看到真正删除之后被删除的记录加入垃圾链表的头结点,修改垃圾链表PAGE_FREE属性的值。

⻚⾯的Page Header部分有⼀个PAGE_GARBAGE属性,该属性记录着当前⻚⾯中可重⽤存储空间占⽤的总字节数。每当有已删除记录被加⼊到垃圾链表后,都会把这个PAGE_GARBAGE属性的值加上该已删除记录占⽤的存储空间⼤⼩。PAGE_FREE指向垃圾链表的头节点,之后每当新插⼊记录时,⾸先判断PAGE_FREE指向的头节点代表的已删除记录占⽤的存储空间是否⾜够容纳这条新插⼊的记录,如果不可以容纳,就直接向⻚⾯中申请新的空间来存储这条记录(是的,你没看错,并不会尝试遍历整个垃圾链表,找到⼀个可以容纳新记录的节点)。如果可以容纳,那么直接重⽤这条已删除记录的存储空间,并且把PAGE_FREE指向垃圾链表中的下⼀条已删除记录。但是
这⾥有⼀个问题,如果新插⼊的那条记录占⽤的存储空间⼤⼩⼩于垃圾链表的头节点占⽤的存储空间⼤⼩,那就意味头节点对应的记录占⽤的存储空间⾥有⼀部分空间⽤不到,这部分空间就被称之为碎⽚空间。那这些碎⽚空间岂不是永远都⽤不到了么?其实也不是,这些碎⽚空间占⽤的存储空间⼤⼩会被统计到PAGE_GARBAGE属性中,这些碎⽚空间在整个⻚⾯快使⽤完前并不会被重新利⽤,不过当⻚⾯快满时,如果再插⼊⼀条记录,此时⻚⾯中并不能分配⼀条完整记录的空间,这时候会⾸先看⼀看PAGE_GARBAGE的空间和剩余可利⽤的空间加起来是不是可以容纳下这条记录,如果可以的话,InnoDB会尝试重新组织⻚内的记录,重新组织的过程就是先开辟⼀个临时⻚⾯,把⻚⾯内的记录依次插⼊⼀遍,因为依次插⼊时并不会产⽣碎⽚,之后再把临时⻚⾯的内容复制到本⻚⾯,这样就可以把那些碎⽚空间都解放出来(很显然重新组织⻚⾯内的记录⽐较耗费性能)。
不知道大家意识到没有,在删除语句所在的事务提交之前,只会经历阶段一,也就是delete mark阶段,提交之后我们就不用回滚,所以只需要考虑删除操作阶段一做的影响进行回滚。InnoDB为阶段一设计了 TRX_UNDO_DEL_MARK_REC 类型的 undo ⽇志

undo日志使用的ibp,到底有没有undo buffer?

undo日志写在磁盘的系统表空间,即数据目录的ibdata1文件中,是字扩展文件

undo日志刷入磁盘的时机,在redo日志之后,因为undo日志也是写在了页上面当然不是数据页,而是其他类型的页,所以在写页之前肯定会先刷入redo日志。

undo页其实在事务提交之后就没用了,就可以删除了。

参考:

https://blog.51cto.com/u_16099245/8284858?u_atoken=cb0a23c68bb36e0b51bdd6698c0d1b1e&u_asession=010oqemvtD-SI70HvdIIJ3SD6rL-0g4lqUd_UCHB5kIzJd_RzP7PK5M9OMtfYEHkSMdlmHJsN3PcAI060GRB4YZGyPlBJUEqctiaTooWaXr7I&u_asig=05KSFHoAoEbI0nFMyiGbILePlRrb4fDSnENTW1oe0zxFMc_aeYEexCgEgeGZ16H7ZE1DUDCzmL2emmn6qmFC9CC_XK8dYD1h4eorn2e4KHNprayj0S6sDcNlfu97C_4rOSwYUPHbo6E_XIjuXpGnuIY3njx0LLgd3IEivFU82Koe5g2QMxYs6lyXb1lFWKql56bKAECQJI7JT5rwB3i3DMxmVX6G2FWvrtzLd69hasdI_jtcevR-HxX_TJmPgBpiSXfv5D2tGhDz888pbf-PhVCPH6b79Mch4xN3KRAVxzONmsTpJ-4hEVCCqo-GZeD3WUZHi7af-9T9DT_5BT1SiXZw&u_aref=KA7pqKy7uBwsbhILbK67J7v6BqM%3D

MySQL 的undo什么时候写到硬盘? - 知乎

003_Mysql之事务详解redo log、undolog 和 MVCC_undolog存放在磁盘的哪里-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/703921.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI 绘画:人工智能绘画之美

人工智能(AI)是当今科技领域的热门话题,它不仅可以帮助我们解决各种复杂的问题,还可以创造出令人惊叹的艺术作品。AI 绘画是一种利用 AI 技术生成图像的方法,它可以模仿不同的风格、主题和技巧,甚至可以创造…

Qt Linux下调用OpenGL的glu.h报错:error: GL/glu.h: No such file or directory

Qt Linux下调用OpenGL的glu.h报错:error: GL/glu.h: No such file or directory 引言一、问题描述二、解决方案三、解决过程记录3.1 定位问题3.2 尝试使用yum命令安装3.3 从网上下载到本地进行安装 引言 在Windows上正常运行的OpenGL程序,到Linux下突然…

cuda学习笔记(2)

一 专业名词 1 分支断定 2 一致性和同一性 3 常见名词汇总 4 加速比 二 GPU架构构述 GPU就是将cpu的数据存储单元去掉,也就是保留执行单元,GPU就是多个执行单元 1 GPU设计思路,指令流共享,同时执行,数据切分成小块 …

四种主流的prompt框架

省流版: 文章介绍了在使用GPT时的四种prompt框架,有利于使用者打磨提问风格,与GPT进行更好的交互以提高生产力,能帮助大家有效提高工作效率~ 创作不易,如果对你有帮助的话,还请三连支持~ 想要使用Prompt…

MySQL的21个SQL经验

1. 写完SQL先explain查看执行计划(SQL性能优化) 日常开发写SQL的时候,尽量养成这个好习惯呀:写完SQL后,用explain分析一下,尤其注意走不走索引。 explain select userid,name,age from user where userid =10086 or age =18;2、操作delete或者update语句,加个limit(S…

jQuery简介与解析 - 掌控网页互动的魔法工具

jQuery简介与解析 - 掌控网页互动的魔法工具 摘要:本文将带您了解jQuery这一强大且流行的JavaScript库,探讨其特点、优势以及如何在网页开发中发挥巨大作用。我们将从jQuery的基本概念入手,逐步深入解析其核心功能,助您轻松掌握这…

phpspreadsheet导出数据和图片到excel

仅作记录&#xff0c;废话不多说 前提是已经安装了phpspreadsheet &#xff08; composer require phpoffice/phpspreadsheet &#xff09; 一、 数据拼装&#xff0c;调用excel类 <?php /*** 电子台账* Date: 2023/4/20* Time: 17:28*/namespace app\store\controlle…

Android 面试问题 2024 版(其三)

Android 面试问题 2024 版&#xff08;其三&#xff09; 十一、版本控制十二、Play 商店和应用程序部署十三、无障碍十四、第三方库和 API十五、解决问题的能力十六、基于 JD 的非常高级别的问题 十一、版本控制 什么是版本控制&#xff0c;为什么它在软件开发中很重要&#x…

Shell echo、printf、test命令

目录 Shell echo命令 打印文本消息 显示变量值 输出特殊字符 输出到文件 追加到文件 Shell printf 命令 打印简单文本 Shell test 命令 文件测试 字符串比较 整数比较 逻辑运算 Shell echo命令 打印文本消息 echo "Hello, World!" 显示变量值 name&q…

积累:如何提取 int 数据的高低字节

前言 服务通信、硬件开发&#xff0c;一般都会涉及到字节数据的解析、处理。无论是两个服务间的数据交互协议&#xff0c;还是硬件设备的通信协议&#xff0c;协议中涉及到的参数比较多&#xff0c;并且协议中每个参数占用的字节大小设定会因功能也不尽相同&#xff1a;简单点的…

微信小程序(4)- 事件系统和模板语法

1. 事件系统 1.1 事件绑定和事件对象 小程序中绑定事件与在网页开发中绑定事件几乎一致&#xff0c;只不过在小程序不能通过 on 的方式绑定事件&#xff0c;也没有 click 等事件&#xff0c;小程序中绑定事件使用 bind 方法&#xff0c;click 事件也需要使用 tap 事件来进行代…

抖店货源怎么找?这几个货源渠道,我都替你整理出来了!

我是电商珠珠 在开通抖店之后&#xff0c;怎么找货源成为了新手的致命要点。货源找不好&#xff0c;就会导致店铺的流量曝光不够。 抖店货源究竟该怎么找呢&#xff0c;今天我就来给大家说个明白。 1、货源网站 比较常规的方式&#xff0c;就是去货源网站上去找&#xff0c…

LeetCode //C - 131. Palindrome Partitioning

131. Palindrome Partitioning Given a string s, partition s such that every substring of the partition is a palindrome. Return all possible palindrome partitioning of s. Example 1: Input: s “aab” Output: [[“a”,“a”,“b”],[“aa”,“b”]] Example 2…

达梦数据库——集群守护进程各状态详解

守护进程&#xff08; dmwatcher &#xff09;是 DM 数据守护系统不可或缺的核心部件&#xff0c;是数据库实例和 监视器之间信息流转的桥梁。数据库实例向本地守护进程发送信息&#xff0c;接收本地守护进程的消 息和命令&#xff1b;监视器&#xff08; dmmonitor &#…

爬某网站延禧宫率第一集视频

import requests import re from tqdm import tqdm # 网址链接&#xff1a;https://v.ijujitv.cc/play/24291-1-1.html url https://v6.1080pzy.co/20220801/urxniJCN/hls/index.m3u8 headers {"User-Agent":"Mozilla/5.0 (Windows NT 10.0; Win64; x64) Appl…

BUUCTF crypto做题记录(9)新手向

一、rsa2 得到题目代码如下&#xff1a; N 101991809777553253470276751399264740131157682329252673501792154507006158434432009141995367241962525705950046253400188884658262496534706438791515071885860897552736656899566915731297225817250639873643376310103992170…

【Spring连载】使用Spring Data的Repositories----从聚合根(Aggregate Roots)发布事件

【Spring连载】使用Spring Data的Repositories----从聚合根Aggregate Roots发布事件 由存储库管理的实体是聚合根。在域驱动设计应用程序中&#xff0c;这些聚合根通常发布域事件。Spring Data提供了一个名为DomainEvents的注解&#xff0c;你可以在聚合根的方法上使用该注解&a…

Linux调用可执行程序:system()函数和execl函数

system()函数&#xff1a; system()函数是一个在C/C编程语言中的库函数&#xff0c;用于在操作系统中执行命令。 函数声明如下&#xff1a; int system(const char *command);该函数接受一个指向以空字符结尾的字符串的指针作为参数&#xff0c;该字符串包含要执行的命令。函…

吴恩达deeplearning.ai:sigmoid函数的替代方案以及激活函数的选择

以下内容有任何不理解可以翻看我之前的博客哦&#xff1a;吴恩达deeplearning.ai专栏 文章目录 引入——改进下需求预测模型ReLU函数(整流线性单元 rectified linear unit&#xff09;线性激活函数(linear activation function)激活函数的选择实现方式为什么需要激活函数 到现在…

【MATLAB】 LMD信号分解+FFT傅里叶频谱变换组合算法

有意向获取代码&#xff0c;请转文末观看代码获取方式~ 展示出图效果 1 LMD分解算法 LMD (Local Mean Decomposition) 分解算法是一种信号分解算法&#xff0c;它可以将一个信号分解成多个局部平滑的成分&#xff0c;并且可以将高频噪声和低频信号有效地分离出来。LMD 分解算…