MySQL日志UndoLog的作用

 更新时间:2025年12月09日 11:22:06   作者:猩火燎猿  
UndoLog是InnoDB用于事务回滚和MVCC的重要机制,本文主要介绍了MySQL日志UndoLog的作用,文中介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、Undo Log 的作用

  1. 事务回滚
    当事务执行过程中出现错误或主动回滚时,InnoDB 需要将数据恢复到修改前的状态。Undo Log 记录了数据被修改前的信息,可以根据 Undo Log 回滚数据。

  2. MVCC(多版本并发控制)
    为了支持高并发下的读写,InnoDB 实现了 MVCC。Undo Log 保存了数据修改前的版本信息,快照读(如 SELECT … WHERE …)可以通过 Undo Log 读取到历史版本的数据,实现非阻塞读。

  3. 崩溃恢复
    在数据库异常宕机时,Undo Log 可以帮助恢复未提交事务的数据,保证数据一致性。

二、Undo Log 的分类

Undo Log 主要分为两种类型:

  1. Insert Undo Log
    记录插入操作(INSERT),主要用于回滚未提交的插入操作。
    只在事务未提交前有效,事务提交后会被立即清理。

  2. Update Undo Log
    记录更新操作(UPDATE、DELETE),保存被修改行的旧版本。
    除了回滚,还用于 MVCC 的快照读,因此即使事务提交,Update Undo Log 还会保留一段时间,直到没有快照读引用它。

三、Undo Log 的存储

  • 在 InnoDB 中,Undo Log 并不是单独的文件,而是存储在特殊的表空间中(如 undo tablespace)。
  • 早期版本(MySQL 5.7 及之前)Undo Log 存储在系统表空间(ibdata1)中。
  • MySQL 8.0 开始,支持独立的 Undo 表空间,可以更灵活地管理和清理 Undo Log。

四、Undo Log 的工作流程

  1. 当事务对数据进行修改(如 UPDATE),InnoDB 会先将原数据写入 Undo Log。
  2. 事务提交前,Undo Log 会一直保留,用于回滚和快照读。
  3. 事务提交后,Insert Undo Log 会被立即清理;Update Undo Log 会等到没有快照读引用它时才被清理(称为 Purge)。
  4. Undo Log 的清理由后台线程(Purge)异步进行,避免影响主业务。

五、Undo Log 与 MVCC 的关系

  • InnoDB 的 MVCC 依赖 Undo Log 实现。
  • 每行记录都有隐藏字段(如 trx_id),记录最后一次修改的事务ID。
  • 快照读时,InnoDB 会根据 Undo Log 回溯到符合条件的版本,实现非阻塞读。

六、Undo Log 的性能与管理

  • Undo Log 过多会导致表空间膨胀,影响性能。
  • 定期清理(Purge)很重要,MySQL 8.0 支持多线程 Purge,提高清理效率。
  • 可以通过参数配置 Undo 表空间数量、大小等(如 innodb_undo_tablespaces)。

七、相关参数与命令

  • innodb_undo_tablespaces:设置 Undo 表空间数量。
  • innodb_undo_log_truncate:开启自动截断 Undo Log。
  • SHOW ENGINE INNODB STATUS; 可以查看当前 Undo Log 相关信息。

八、常见问题

  1. Undo Log 占用空间过大怎么办?

    • 检查是否有长事务或长时间未提交的快照读。
    • 增加 Purge 线程数(innodb_purge_threads)。
    • 合理设计业务,避免长事务。
  2. Undo Log 会影响性能吗?

    • Undo Log 写入是必须的,但一般不会成为性能瓶颈。
    • 长时间未清理的 Undo Log 可能导致空间膨胀,影响性能。

九、Undo Log 的结构

Undo Log 并不是简单地记录一行数据的旧值,它实际上是一个链式的数据结构。每个数据行的修改都会生成一个 Undo Log 记录,这些记录通过链表串联起来,形成所谓的“版本链”。

  • Undo Record
    每个 Undo Log 记录包含:

    • 被修改的行主键
    • 修改前的列值
    • 事务ID(trx_id)
    • 操作类型(Insert、Update、Delete)
    • 指向上一个版本的指针
  • 版本链(Version Chain)
    每行数据都可以通过 Undo Log 找到历史版本。快照读时,InnoDB 会沿着版本链查找满足可见性规则的版本。

十、Undo Log 的生命周期

  1. 生成
    事务对数据进行更改时,立即生成 Undo Log 记录。

  2. 保留

    • 未提交事务:Undo Log 必须保留,以便回滚。
    • 已提交事务:Update Undo Log 还需保留,直到所有快照读都不再需要该版本。
  3. 清理(Purge)

    • 后台 Purge 线程会定期清理不再被引用的 Undo Log。
    • 清理时,Undo Log 会从表空间中物理删除。

十一、Undo Log 的清理机制(Purge)

  • 触发时机
    1. 事务提交后,Insert Undo Log 可立即清理。
    2. Update Undo Log 只有在所有快照读完成后才能清理。
  • 清理过程
    • InnoDB 的 Purge 线程会扫描 Undo Log 链,判断是否有快照读还需要旧版本。
    • 如果没有引用,Undo Log 被物理删除,释放空间。
  • 参数调优
    • innodb_purge_threads:设置 Purge 线程数量,提升清理效率。
    • innodb_max_purge_lag:控制 Purge 的延迟,防止主业务受影响。

十二、Undo Log 与 Redo Log 的区别

方面Undo LogRedo Log
作用回滚、MVCC、快照读崩溃恢复、持久化
写入时机数据修改前数据修改后
内容记录数据修改前的旧值记录数据修改后的新值
事务提交时处理部分立即清理,部分延迟清理需要持久化写盘
存储位置Undo 表空间Redo 日志文件(ib_logfile)

十三、典型问题分析

  1. 长事务导致 Undo Log 堆积

    • 长事务或长时间未提交的快照读会导致 Undo Log 无法清理,表空间膨胀。
    • 解决方法:优化业务逻辑,避免长事务,及时提交。
  2. Undo Log 影响 DDL 操作

    • 大量 Undo Log 会影响表的 DDL(如 ALTER TABLE),因为 DDL 需要等待相关 Undo Log 清理。
    • 解决方法:在执行 DDL 前,确保没有长事务和大批量快照读。
  3. Undo Log 空间回收慢

    • Undo Log 清理依赖 Purge 线程,线程数不足会导致清理慢。
    • 解决方法:适当增加 innodb_purge_threads,观察 SHOW ENGINE INNODB STATUS

十四、Undo Log 相关源码简析

(如需更深入源码分析可继续追问)

  • InnoDB 的 Undo Log 相关代码主要在 trx0undo.ctrx0purge.crow0vers.c 等文件。
  • 源码实现了 Undo Log 的生成、链表管理、清理、与 MVCC 的交互。

十五、实践建议

  1. 避免长事务和大批量未提交的快照读。
  2. 合理配置 Undo 表空间数量和大小。
  3. 关注 Purge 线程运行状况,及时调优相关参数。
  4. 定期监控 Undo Log 的空间占用和清理效率。
  5. 在大数据量操作或 DDL 前,提前处理相关事务,确保 Undo Log 不会堆积。

总结

Undo Log 是 InnoDB 实现事务、回滚、MVCC 的核心机制。合理管理 Undo Log,有助于提升数据库的稳定性和性能。

到此这篇关于MySQL日志UndoLog的作用的文章就介绍到这了,更多相关MySQL日志UndoLog内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • MySQL在Centos7环境安装的完整步骤记录

    MySQL在Centos7环境安装的完整步骤记录

    在CentOS7环境下安装MySQL是一项常见的任务,尤其对于那些没有网络连接或者需要在隔离环境中的开发者来说,离线安装MySQL显得尤为重要,这篇文章主要介绍了MySQL在Centos7环境安装的完整步骤,需要的朋友可以参考下
    2024-10-10
  • 简介操作MySQL日志的一些方法

    简介操作MySQL日志的一些方法

    这篇文章主要介绍了操作MySQL日志的一些方法,对日志的删除操作做了重点讲解,需要的朋友可以参考下
    2015-07-07
  • MySQL压力测试工具Mysqlslap的使用

    MySQL压力测试工具Mysqlslap的使用

    这篇文章主要介绍了MySQL官方压力测试工具 Mysqlslap的使用方法,帮助大家更好的理解和使用MySQL,感兴趣的朋友可以了解下
    2020-12-12
  • MySQL Workbench快速引入sql文件的图文教程

    MySQL Workbench快速引入sql文件的图文教程

    在MySQL使用时,通常需要将数据库sql文件保存,或者是导入他人的sql文件,可以使用workbench处理,这篇文章主要给大家介绍了关于MySQL Workbench快速引入sql文件的相关资料,需要的朋友可以参考下
    2023-11-11
  • MySQL窗口函数实现榜单排名

    MySQL窗口函数实现榜单排名

    相信大家在日常的开发中经常会碰到榜单类的活动需求,本文主要介绍了MySQL窗口函数实现榜单排名,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • Mysql数据库百万级数据测试索引效果

    Mysql数据库百万级数据测试索引效果

    这篇文章主要为大家介绍了Mysql数据库百万数据测试索引效果,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • linux mysql忘记密码的多种解决或Access denied for user ''root''@''localhost''

    linux mysql忘记密码的多种解决或Access denied for user ''root''@''local

    linux mysql忘记密码的多种解决方法。
    2009-07-07
  • k8s上运行的mysql、mariadb数据库的备份记录(支持x86和arm两种架构)

    k8s上运行的mysql、mariadb数据库的备份记录(支持x86和arm两种架构)

    本文记录在K8s上运行的MySQL/MariaDB备份方案,通过工具容器执行mysqldump,结合定时任务实现自动备份,支持X86和ARM架构,并强调cron环境需转义%符号及避免使用-it参数,对k8s mysql、mariadb数据库备份步骤感兴趣的朋友一起看看吧
    2025-06-06
  • mysql唯一性约束unique代码示例

    mysql唯一性约束unique代码示例

    这篇文章主要介绍了mysql唯一性约束unique的相关资料,需要的朋友可以参考下
    2025-11-11
  • MySQL查看表的最后一个ID的常见方法

    MySQL查看表的最后一个ID的常见方法

    在使用MySQL数据库时,我们经常会遇到需要查看表中最后一个id值的场景,无论是为了调试、数据分析还是其他用途,了解如何快速获取最后一个id都是非常实用的技能,本文将介绍几种常见的方法,帮助你在不同场景下轻松找到表中的最后一个id,需要的朋友可以参考下
    2025-05-05

最新评论