MySQL数库多久刷一次盘?
前言表数据刷什么到磁盘什么时候刷到磁盘谁来负责刷盘前言
在 Buffer Pool 中被修改过的 Page(页)都会被标记成脏页,放到一个链表(Flush 链表)里。当脏页的数量达到了 Buffer Pool 中页数量的 **10%,当然 10% 这个值是可变的,通过配置项innodb_max_dirty_pages_pct_lwm 来配置的,其默认值为 10%,并且这个值也必须小于另一个配置innodb_max_dirty_pages_pct 的值(90%**)。 至于启动多少个线程,则是由另一个变量 innodb_page_cleaners来控制的,默认是 4.一般都不会去改这个。
表数据
我们这篇「短文」讨论的是【MySQL 表数据多久刷一次盘】,从这个标题中我们可以分为两个问题:
1、把什么刷到磁盘中?
2、什么时候刷到磁盘中? 把什么刷到磁盘中
看上去有点废话,肯定是将数据刷入磁盘。所以我们更多需要讨论的是【数据是以什么样的形式被刷入磁盘】。 答案是页
在 InnoDB 中, 页 是数据被管理的最小的单位。当使用 InnoDB 作为存储引擎的 MySQL 运行时,表中一行一行的数据会被组织在一页一页当中,放在 Buffer Pool 中。
这一页一页的数据,就存放在 Buffer Pool 中。当 DML 语句(也就是 CRUD)语句对表数据进行了变更之后,数据所在的那一页就会被标记为 脏页 。
InnoDB 会用一个叫【Flush 链表】的结构来存放这些脏页,凡是被放进该链表的页都代表需要 刷入磁盘 ,但不是立即刷入。
和 InnoDB 的其他日志例如 Redo Log 一样,这些日志都是有自己的 刷盘策略 。例如 Redo Log,其刷盘策略可以用下图来表示:
1、参数为0,Redo Log 会每隔一秒,写入并且刷入磁盘。 2、参数为1,Redo Log 会在每次事务提交之后刷入磁盘 3、参数为2,每次事务提交,都会写到 OS 缓存中去,然后每隔一秒将 OS 缓存中的数据刷入磁盘
参考资料:Innodb内存结构和磁盘结构什么时候刷到磁盘1、接上节,策略就是:脏页的数量达到了 Buffer Pool 中页数量的 **10%**,就会触发将 Flush 链表中的 脏页刷入磁盘。举个例子,Buffer Pool 中总共有 100 张页,脏页如果达到了 10 页就会启动后台线程, 触发刷盘。 2、当然,【10%】这个数值是可配置的,通过 MySQL 配置项 innodb_max_dirty_pages_pct_lwm 可以 进行调整,只是默认值是 10%。但是我们调整的值不能超过某个最大值,这个最大值由 innodb_max_dirty_pages_pct 来指定,默认值为 90%。 3、换句话说,默认情况,刷盘阈值是 10%,如果需要自定义,则最大值不能超过 90%。谁来负责刷盘
上个小节已经说过了,会启动线程来专门做这个事情,这个没有什么疑问。我们需要关注的是会启动多少个线程来做这个事。答案是 4 个,我们也可以通过配置项 innodb_page_cleaners 来更改,但一般都不会去改这个值。
刷盘线程可以参考:Innodb后台线程