写点什么

MySQL 不会丢失数据的秘密,就藏在它的 7 种日志里

发布于: 2021 年 01 月 07 日
MySQL不会丢失数据的秘密,就藏在它的 7种日志里

本文收录在 GitHub 地址 https://github.com/chengxy-nds/Springboot-Notebook


进入正题前先简单看看 MySQL 的逻辑架构,相信我用的着。



MySQL 的逻辑架构大致可以分为三层:


  • 第一层:处理客户端连接、授权认证,安全校验等。


  • 第二层:服务器server层,负责对 SQL 解释、分析、优化、执行操作引擎等。


  • 第三层:存储引擎,负责 MySQL 中数据的存储和提取。


>我们要知道 MySQL 的服务器层是不管理事务的,事务是由存储引擎实现的,而 MySQL 中支持事务的存储引擎又属InnoDB使用的最为广泛,所以后续文中提到的存储引擎都以InnoDB为主。



记住! *记住!* 记住! 上边这张图,她是 MySQL 更新数据的基础流程,其中包括redo logbin logundo log三种日志间的大致关系,好了闲话少说直奔主题。


redo log(重做日志)


redo log属于 MySQL 存储引擎InnoDB的事务日志。


MySQL 的数据是存放在磁盘中的,每次读写数据都需做磁盘 IO 操作,如果并发场景下性能就会很差。为此 MySQL 提供了一个优化手段,引入缓存Buffer Pool。这个缓存中包含了磁盘中**部分**数据页(page)的映射,以此来缓解数据库的磁盘压力。


当从数据库读数据时,首先从缓存中读取,如果缓存中没有,则从磁盘读取后放入缓存;当向数据库写入数据时,先向缓存写入,此时缓存中的数据页数据变更,这个数据页称为脏页Buffer Pool中修改完数据后会按照设定的更新策略,定期刷到磁盘中,这个过程称为刷脏页


MySQL 宕机


如果刷脏页还未完成,可 MySQL 由于某些原因宕机重启,此时Buffer Pool中修改的数据还没有及时的刷到磁盘中,就会导致数据丢失,无法保证事务的持久性。


为了解决这个问题引入了redo log,redo Log 如其名侧重于重做!它记录的是数据库中每个页的修改,而不是某一行或某几行修改成怎样,可以用来恢复提交后的物理数据页,且只能恢复到最后一次提交的位置。


redo log用到了WAL(Write-Ahead Logging)技术,这个技术的核心就在于修改记录前,一定要先写日志,并保证日志先落盘,才能算事务提交完成。


有了 redo log 再修改数据时,InnoDB 引擎会把更新记录先写在 redo log 中,在修改Buffer Pool中的数据,当提交事务时,调用fsync把 redo log 刷入磁盘。至于缓存中更新的数据文件何时刷入磁盘,则由后台线程异步处理。


注意:此时 redo log 的事务状态是prepare,还未真正提交成功,要等bin log日志写入磁盘完成才会变更为commit,事务才算真正提交完成。


这样一来即使刷脏页之前 MySQL 意外宕机也没关系,只要在重启时解析 redo log 中的更改记录进行重放,重新刷盘即可。


大小固定


redo log 采用固定大小,循环写入的格式,当 redo log 写满之后,重新从头开始如此循环写,形成一个环状。


那为什么要如此设计呢?


因为 redo log 记录的是数据页上的修改,如果Buffer Pool中数据页已经刷磁盘后,那这些记录就失效了,新日志会将这些失效的记录进行覆盖擦除。



上图中的write pos表示 redo log 当前记录的日志序列号LSN(log sequence number),写入还未刷盘,循环往后递增;check point表示 redo log 中的修改记录已刷入磁盘后的 LSN,循环往后递增,这个 LSN 之前的数据已经全落盘。


write poscheck point之间的部分是 redo log 空余的部分(绿色),用来记录新的日志;check pointwrite pos之间是 redo log 已经记录的数据页修改数据,此时数据页还未刷回磁盘的部分。当write pos追上check point时,会先推动check point向前移动,空出位置(刷盘)再记录新的日志。


注意:redo log 日志满了,在擦除之前,需要确保这些要被擦除记录对应在内存中的数据页都已经刷到磁盘中了。擦除旧记录腾出新空间这段期间,是不能再接收新的更新请求的,此刻 MySQL 的性能会下降。所以在并发量大的情况下,合理调整 redo log 的文件大小非常重要。


crash-safe


因为 redo log 的存在使得Innodb引擎具有了crash-safe的能力,即 MySQL 宕机重启,系统会自动去检查 redo log,将修改还未写入磁盘的数据从 redo log 恢复到 MySQL 中。


MySQL 启动时,不管上次是正常关闭还是异常关闭,总是会进行恢复操作。会先检查数据页中的LSN,如果这个 LSN 小于 redo log 中的 LSN,即write pos位置,说明在redo log上记录着数据页上尚未完成的操作,接着就会从最近的一个check point出发,开始同步数据。


简单理解,比如:redo log 的LSN是 500,数据页的LSN是 300,表明重启前有部分数据未完全刷入到磁盘中,那么系统则将 redo log 中LSN序号 300 到 500 的记录进行重放刷盘。



undo log(回滚日志)


undo log也是属于 MySQL 存储引擎 InnoDB 的事务日志。


undo log属于逻辑日志,如其名主要起到回滚的作用,它是保证事务原子性的关键。记录的是数据修改前的状态,在数据修改的流程中,同时会记录一条与当前操作相反的逻辑日志到undo log中。


我们举个栗子:假如更新 ID=1 记录的 name 字段,name 原始数据为小富,现改 name 为程序员内点事


事务执行update X set name = 程序员内点事 where id =1语句时,先会在undo log中记录一条相反逻辑的update X set name = 小富 where id =1记录,这样当某些原因导致服务异常事务失败,就可以借助undo log将数据回滚到事务执行前的状态,保证事务的完整性。



那可能有人会问:同一个事物内的一条记录被多次修改,那是不是每次都要把数据修改前的状态都写入undo log呢?


答案是不会的!


undo log只负责记录事务开始前要修改数据的原始版本,当我们再次对这行数据进行修改,所产生的修改记录会写入到redo logundo log负责完成回滚,redo log负责完成前滚。


回滚


未提交的事务,即事务未执行commit。但该事务内修改的脏页中,可能有一部分脏块已经刷盘。如果此时数据库实例宕机重启,就需要用回滚来将先前那部分已经刷盘的脏块从磁盘上撤销。


前滚


未完全提交的事务,即事务已经执行commit,但该事务内修改的脏页中只有一部分数据被刷盘,另外一部分还在buffer pool缓存上,如果此时数据库实例宕机重启,就需要用前滚来完成未完全提交的事务。将先前那部分由于宕机在内存上的未来得及刷盘数据,从redo log中恢复出来并刷入磁盘。


数据库实例恢复时,先做前滚,后做回滚。


如果你仔细看过了上边的 MySQL数据更新流程图 就会发现,undo logredo logbin log三种日志都是在刷脏页之前就已经刷到磁盘了的,相互协作最大限度保证了用户提交的数据不丢失。


bin log(归档日志)


bin log是一种数据库 Server 层(和什么引擎无关),以二进制形式存储在磁盘中的逻辑日志。bin log记录了数据库所有DDLDML操作(不包含 SELECTSHOW等命令,因为这类操作对数据本身并没有修改)。


默认情况下,二进制日志功能是关闭的。可以通过以下命令查看二进制日志是否开启:


mysql> SHOW VARIABLES LIKE 'log_bin';+---------------+-------+| Variable_name | Value |+---------------+-------+| log_bin       | OFF   |+---------------+-------+
复制代码


bin log也被叫做归档日志,因为它不会像redo log那样循环写擦除之前的记录,而是会一直记录日志。一个bin log日志文件默认最大容量1G(也可以通过maxbinlogsize参数修改),单个日志超过最大值,则会新创建一个文件继续写。


mysql> show binary logs;+-----------------+-----------+| Log_name        | File_size |+-----------------+-----------+| mysq-bin.000001 |      8687 || mysq-bin.000002 |      1445 || mysq-bin.000003 |      3966 || mysq-bin.000004 |       177 || mysq-bin.000005 |      6405 || mysq-bin.000006 |       177 || mysq-bin.000007 |       154 || mysq-bin.000008 |       154 |
复制代码


bin log日志的内容格式其实就是执行 SQL 命令的反向逻辑,这点和undo log有点类似。一般来说开启bin log都会给日志文件设置过期时间(expirelogsdays参数,默认永久保存),要不然日志的体量会非常庞大。


mysql> show variables like 'expire_logs_days';+------------------+-------+| Variable_name    | Value |+------------------+-------+| expire_logs_days | 0     |+------------------+-------+1 row in set
mysql> SET GLOBAL expire_logs_days=30;Query OK, 0 rows affected
复制代码


bin log主要应用于 MySQL 主从模式(master-slave)中,主从节点间的数据同步;以及基于时间点的数据还原。


主从同步


通过下图 MySQL 的主从复制过程,来了解下bin log在主从模式下的应用。



  • 用户在主库master执行DDLDML操作,修改记录顺序写入bin log;


  • 从库slave的 I/O 线程连接上 Master,并请求读取指定位置position的日志内容;


  • Master收到从库slave请求后,将指定位置position之后的日志内容,和主库 bin log 文件的名称以及在日志中的位置推送给从库;


  • slave 的 I/O 线程接收到数据后,将接收到的日志内容依次写入到relay log文件最末端,并将读取到的主库 bin log 文件名和位置position记录到master-info文件中,以便在下一次读取用;


  • slave 的 SQL 线程检测到relay log中内容更新后,读取日志并解析成可执行的 SQL 语句,这样就实现了主从库的数据一致;


基于时间点还原


我们看到bin log也可以做数据的恢复,而redo log也可以,那它们有什么区别?


  • 层次不同:redo log 是 InnoDB 存储引擎实现的,bin log 是 MySQL 的服务器层实现的,但 MySQL 数据库中的任何存储引擎对于数据库的更改都会产生 bin log。


  • 作用不同:redo log 用于碰撞恢复(crash recovery),保证 MySQL 宕机也不会影响持久性;bin log 用于时间点恢复(point-in-time recovery),保证服务器可以基于时间点恢复数据和主从复制。


  • 内容不同:redo log 是物理日志,内容基于磁盘的页Page;bin log 的内容是二进制,可以根据binlog_format参数自行设置。


  • 写入方式不同:redo log 采用循环写的方式记录;binlog 通过追加的方式记录,当文件大小大于给定值后,后续的日志会记录到新的文件上。


  • 刷盘时机不同:bin log 在事务提交时写入;redo log 在事务开始时即开始写入。


bin log 与 redo log 功能并不冲突而是起到相辅相成的作用,需要二者同时记录,才能保证当数据库发生宕机重启时,数据不会丢失。


relay log(中继日志)


relay log日志文件具有与bin log日志文件相同的格式,从上边 MySQL 主从复制的流程可以看出,relay log起到一个中转的作用,slave先从主库master读取二进制日志数据,写入从库本地,后续再异步由SQL线程读取解析relay log为对应的 SQL 命令执行。


slow query log


慢查询日志(slow query log): 用来记录在 MySQL 中执行时间超过指定时间的查询语句,在 SQL 优化过程中会经常使用到。通过慢查询日志,我们可以查找出哪些查询语句的执行效率低,耗时严重。


出于性能方面的考虑,一般只有在排查慢 SQL、调试参数时才会开启,默认情况下,慢查询日志功能是关闭的。可以通过以下命令查看是否开启慢查询日志:


mysql> SHOW VARIABLES LIKE 'slow_query%';+---------------------+--------------------------------------------------------+| Variable_name       | Value                                                  |+---------------------+--------------------------------------------------------+| slow_query_log      | OFF                                                    || slow_query_log_file | /usr/local/mysql/data/iZ2zebfzaequ90bdlz820sZ-slow.log |+---------------------+--------------------------------------------------------+
复制代码


通过如下命令开启慢查询日志后,我发现 iZ2zebfzaequ90bdlz820sZ-slow.log 日志文件里并没有内容啊,可能因为我执行的 SQL 都比较简单没有超过指定时间。


mysql>  SET GLOBAL slow_query_log=ON;Query OK, 0 rows affected
复制代码


上边提到超过 指定时间 的查询语句才算是慢查询,那么这个时间阈值又是多少嘞?我们通过 long_query_time 参数来查看一下,发现默认是 10 秒。


mysql> SHOW VARIABLES LIKE 'long_query_time';+-----------------+-----------+| Variable_name   | Value     |+-----------------+-----------+| long_query_time | 10.000000 |+-----------------+-----------+
复制代码


这里我们将 long_query_time 参数改小为 0.001 秒再次执行查询 SQL,看看慢查询日志里是否有变化。


mysql> SET GLOBAL long_query_time=0.001;Query OK, 0 rows affected
复制代码


果然再执行 SQL 的时,执行时间大于 0.001 秒,发现慢查询日志开始记录了。



general query log


一般查询日志(general query log):用来记录用户的所有操作,包括客户端何时连接了服务器、客户端发送的所有SQL以及其他事件,比如 MySQL 服务启动和关闭等等。MySQL服务器会按照它接收到语句的先后顺序写入日志文件。


由于一般查询日志记录的内容过于详细,开启后 Log 文件的体量会非常庞大,所以出于对性能的考虑,默认情况下,该日志功能是关闭的,通常会在排查故障需获得详细日志的时候才会临时开启。


我们可以通过以下命令查看一般查询日志是否开启,命令如下:


mysql> show variables like 'general_log';+---------------+-------+| Variable_name | Value |+---------------+-------+| general_log   | OFF   |+---------------+-------+
复制代码


下边开启一般查询日志并查看日志存放的位置。


mysql> SET GLOBAL general_log=on;Query OK, 0 rows affected
复制代码


mysql> show variables like 'general_log_file';+------------------+---------------------------------------------------+| Variable_name    | Value                                             |+------------------+---------------------------------------------------+| general_log_file | /usr/local/mysql/data/iZ2zebfzaequ90bdlz820sZ.log |+------------------+---------------------------------------------------+
复制代码


执行一条查询 SQL 看看日志内容的变化。


mysql> select * from t_config;+---------------------+------------+---------------------+---------------------+| id                  | remark     | create_time         | last_modify_time    |+---------------------+------------+---------------------+---------------------+| 1325741604307734530 | 我是广播表 | 2020-11-09 18:06:44 | 2020-11-09 18:06:44 |+---------------------+------------+---------------------+---------------------+
复制代码


我们看到日志内容详细的记录了所有执行的命令、SQL、SQL 的解析过程、数据库设置等等。



error log


错误日志(error log): 应该是 MySQL 中最好理解的一种日志,主要记录 MySQL 服务器每次启动和停止的时间以及诊断和出错信息。


默认情况下,该日志功能是开启的,通过如下命令查找错误日志文件的存放路径。


mysql> SHOW VARIABLES LIKE 'log_error';+---------------+----------------------------------------------------------------+| Variable_name | Value                                                          |+---------------+----------------------------------------------------------------+| log_error     | /usr/local/mysql/data/LAPTOP-UHQ6V8KP.err |+---------------+----------------------------------------------------------------+
复制代码


注意:错误日志中记录的可并非全是错误信息,像 MySQL 如何启动 InnoDB 的表空间文件、如何初始化自己的存储引擎,初始化 buffer pool 等等,这些也记录在错误日志文件中。



总结


MySQL 作为我们工作中最常接触的中间件,熟练使用只算是入门,如果要在简历写上一笔精通,还需要深入了解其内部工作原理,而这 7 种日志也只是深入学习过程中的一个起点,学无止境,兄嘚干就完了!


>整理了几百本各类技术电子书,有需要的同学可以,在我同名公众号回复[ 666 ]自取。技术群快满了,想进的同学可以加我好友,和大佬们一起吹吹技术,期待你的加入


发布于: 2021 年 01 月 07 日阅读数: 1242
用户头像

不积跬步,无以至千里 2018.07.03 加入

公众号-程序员内点事,一个技术传播者

评论 (3 条评论)

发布
用户头像
画图软件是啥呀?很好看的样子
2021 年 01 月 11 日 16:27
回复
用户头像
"一定要先写日志,并保证日志先落盘,才能算事务提交完成。"
事实上redo log并不一定先落盘。
2021 年 01 月 08 日 15:20
回复
写redo log buffer
2021 年 01 月 11 日 10:52
回复
没有更多了
MySQL不会丢失数据的秘密,就藏在它的 7种日志里