【Mysql】InnoDB统计数据的收集(十三)

我们前边在计算查询成本的时候会用到一些统计数据,比如通过 SHOW TABLE STATUS 可以看到关于表的统计数据,通过 SHOW INDEX 可以看到关于某个索引的统计数据,那么这些统计数据是怎么来的呢?本章节将分享 InnoDB 存储引擎的统计数据收集策略,看完后大家就会明白InnoDB 的统计信息为什么只是一个估计值。

两种不同的统计数据存储方式

InnoDB 提供了两种存储统计数据的方式:

  • 永久性的统计数据
    这种统计数据存储在磁盘上,也就是服务器重启之后这些统计数据还在。
  • 非永久性的统计数据
    这种统计数据存储在内存中,当服务器关闭时这些这些统计数据就都被清除掉了,等到服务器重启之后,在某些适当的场景下会重新收集这些统计数据。

MySQL中提供了系统变量 innodb_stats_persistent 来控制存储统计数据的方式。在 MySQL 5.6.6 之前,innodb_stats_persistent 的值默认是 OFF ,也就是 InnoDB 的统计数据默认是存储到内存的,之后的版本中 innodb_stats_persistent 的值默认是 ON ,也就是统计数据默认被存储到磁盘中。
不过 InnoDB 默认是以表为单位来收集和存储统计数据的,也就是说我们可以把一个database中的某些表的统计数据(以及该表的索引统计数据)存储在磁盘上,把另一些表的统计数据存储在内存中。这可以通过在创建和修改表的时候通过指定 STATS_PERSISTENT 属性来指明该表的统计数据存储方式如下:
CREATE TABLE 表名 (…) Engine=InnoDB, STATS_PERSISTENT = (1|0);
ALTER TABLE 表名 Engine=InnoDB, STATS_PERSISTENT = (1|0);
当 STATS_PERSISTENT=1 时,表示把该表的统计数据永久的存储到磁盘上,当 STATS_PERSISTENT=0 时,表示把该表的统计数据临时的存储到内存中。如果在创建表时未指定 STATS_PERSISTENT 属性,那将采用默认的系统变量 innodb_stats_persistent 的值作为该属性的值。

基于磁盘的永久性统计数据

当我们选择把某个表以及该表索引的统计数据存放到磁盘上时,实际上是把这些统计数据存储到了两个表里:

mysql> SHOW TABLES FROM mysql LIKE 'innodb%';
+---------------------------+
| Tables_in_mysql (innodb%) |
+---------------------------+
| innodb_index_stats        |
| innodb_table_stats        |
+---------------------------+
2 rows in set (0.00 sec)

可以看到,这两个表都位于 mysql 系统数据库下边,其中:

  • innodb_table_stats 存储了关于表的统计数据,每一条记录对应着一个表的统计数据。
  • innodb_index_stats 存储了关于索引的统计数据,每一条记录对应着一个索引的一个统计项的统计数据。

我们下边的任务就是看一下这两个表里边都有什么以及表里的数据是如何生成的。

innodb_table_stats

可以先查看innodb_table_stats这个表中的前3行数据:

mysql> select * from mysql.innodb_table_stats limit 3;
+---------------+---------------+---------------------+--------+----------------------+--------------------------+
| database_name | table_name    | last_update         | n_rows | clustered_index_size | sum_of_other_index_sizes |
+---------------+---------------+---------------------+--------+----------------------+--------------------------+
| charset_demo  | test          | 2023-10-16 22:23:17 |      5 |                    1 |                        0 |
| mysql         | gtid_executed | 2023-10-11 09:43:40 |      0 |                    1 |                        0 |
| sys           | sys_config    | 2023-10-11 09:43:43 |      6 |                    1 |                        0 |
+---------------+---------------+---------------------+--------+----------------------+--------------------------+
3 rows in set (0.00 sec)

各列属性说明:

字段名描述
database_name数据库名
table_name表名
last_update本条记录最后更新时间
n_rows表中记录的条数
clustered_index_size表的聚簇索引占用的页面数量
sum_of_other_index_sizes表的其他索引占用的页面数量

下面查看一下我们之前创建的single_table表的相关信息:

mysql> SELECT * FROM mysql.innodb_table_stats where table_name='single_table';
+---------------+--------------+---------------------+--------+----------------------+--------------------------+
| database_name | table_name   | last_update         | n_rows | clustered_index_size | sum_of_other_index_sizes |
+---------------+--------------+---------------------+--------+----------------------+--------------------------+
| test          | single_table | 2023-11-15 16:00:21 |  10143 |                   97 |                      142 |
+---------------+--------------+---------------------+--------+----------------------+--------------------------+
1 row in set (0.01 sec)

几个重要统计信息项的值如下:

  • n_rows 的值是 10143 ,表明 single_table 表中大约有 10143 条记录,注意这个数据是估计值。
  • clustered_index_size 的值是 97 ,表明 single_table 表的聚簇索引占用97个页面,这个值是也是一个估计值。
  • sum_of_other_index_sizes 的值是 142 ,表明 single_table 表的其他索引一共占用142个页面,这个值是也是一个估计值。

n_rows统计项的收集

我们说过 n_rows 这个统计项的值是估计值,那是为什么呢。 InnoDB 统计一个表中有多少行记录的方法是这样的:

  • 按照一定算法(并不是纯粹随机的)选取几个叶子节点页面,计算每个页面中主键值记录量,然后计算平均一个页面中主键值的记录数量乘以全部叶子节点的数量就算是该表的 n_rows值。
    说明:真实的计算过程比这个稍微复杂一些。但是可以看出来这个 n_rows 值精确与否取决于统计时采样的页面数量,Mysql中有了一个名为 innodb_stats_persistent_sample_pages 的系统变量,用它来控制使用永久性的统计数据时,计算统计数据时采样的页面样本数量。该值设置的越大,统计出的n_rows值越精确,但是统计耗时也就最久;该值设置的越小,统计出的n_rows值越不精确,但是统计耗时少。所以在实际使用是需要我们去权衡,该系统变量的默认值是 20 。

我们知道 InnoDB 默认是以表为单位来收集和存储统计数据的,所以我们也可以单独设置某个表的采样页面的数量,设置方式就是在创建或修改表的时候通过指定STATS_SAMPLE_PAGES 属性来指明该表的统计数据存储方式如下:
CREATE TABLE 表名 (…) Engine=InnoDB, STATS_SAMPLE_PAGES = 具体的采样页面数量;
ALTER TABLE 表名 Engine=InnoDB, STATS_SAMPLE_PAGES = 具体的采样页面数量;
如果我们在创建表的语句中并没有指定 STATS_SAMPLE_PAGES 属性的话,将默认使用系统变量innodb_stats_persistent_sample_pages 的值作为该属性的值。

clustered_index_size和sum_of_other_index_sizes统计项的收集

这两个数据的统计需要使用到我们之前学习的InnoDB 表空间的知识。
这两个统计项的收集过程如下:

  • 从数据字典里找到表的各个索引对应的根页面位置。
    系统表 SYS_INDEXES 里存储了各个索引对应的根页面信息。
  • 从根页面的 Page Header 里找到叶子节点段和非叶子节点段对应的 Segment Header 。
    在每个索引的根页面的 Page Header 部分都有两个字段:
    • PAGE_BTR_SEG_LEAF :表示B+树叶子段的 Segment Header 信息。
    • PAGE_BTR_SEG_TOP :表示B+树非叶子段的 Segment Header 信息。
  • 从叶子节点段和非叶子节点段的 Segment Header 中找到这两个段对应的 INODE Entry 结构。
    这个是 Segment Header 结构:在这里插入图片描述
  • 从对应的 INODE Entry 结构中可以找到该段对应所有零散的页面地址以及 FREE 、 NOT_FULL 、 FULL 链表的基节点。
    这个是 INODE Entry 结构:
    在这里插入图片描述
  • 直接统计零散的页面有多少个,然后从那三个链表的 List Length 字段中读出该段占用的区的大小,每个区占用 64 个页,所以就可以统计出整个段占用的页面。
    这个是链表基节点的示意图:
    在这里插入图片描述
  • 分别计算聚簇索引的叶子结点段和非叶子节点段占用的页面数,它们的和就是clustered_index_size的值,按照同样的方法把其余索引占用的页面数都算出来,加起来之后就是sum_of_other_index_sizes的值。
    这里需要大家注意一个问题,我们说一个段的数据在非常多时(超过32个页面),会以区为单位来申请空间,这里头的问题是以区为单位申请空间中有一些页可能并没有使用,但是在统计 clustered_index_size 和sum_of_other_index_sizes 时都把它们算进去了,所以说聚簇索引和其他的索引占用的页面数可能比这两个值要小一些。

innodb_index_stats

表innodb_index_stats位于mysql这个database下:

mysql> show create table innodb_index_stats;
+--------------------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| Table              | Create Table                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |
+--------------------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
| innodb_index_stats | CREATE TABLE `innodb_index_stats` (
  `database_name` varchar(64) COLLATE utf8_bin NOT NULL,
  `table_name` varchar(199) COLLATE utf8_bin NOT NULL,
  `index_name` varchar(64) COLLATE utf8_bin NOT NULL,
  `last_update` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
  `stat_name` varchar(64) COLLATE utf8_bin NOT NULL,
  `stat_value` bigint(20) unsigned NOT NULL,
  `sample_size` bigint(20) unsigned DEFAULT NULL,
  `stat_description` varchar(1024) COLLATE utf8_bin NOT NULL,
  PRIMARY KEY (`database_name`,`table_name`,`index_name`,`stat_name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin STATS_PERSISTENT=0 |
+--------------------+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

下面看一下这个 innodb_index_stats 表中的各个列的含义:

字段名描述
database_name数据库名
table_name表名
index_name索引名
last_update本条记录最后更新时间
stat_name统计项的名称
stat_value对应的统计项的值
sample_size为生成统计数据而采样的页面数量
stat_description对应的统计项的描述

注意这个表的主键是 (database_name,table_name,index_name,stat_name) ,其中的 stat_name 是指统计项的名
称,也就是说innodb_index_stats表的每条记录代表着一个索引的一个统计项。如下看single_table表的索引数据:

mysql> SELECT * FROM mysql.innodb_index_stats WHERE table_name = 'single_table';
+---------------+--------------+--------------+---------------------+--------------+------------+-------------+-----------------------------------+
| database_name | table_name   | index_name   | last_update         | stat_name    | stat_value | sample_size | stat_description                  |
+---------------+--------------+--------------+---------------------+--------------+------------+-------------+-----------------------------------+
| test          | single_table | PRIMARY      | 2023-10-25 09:56:21 | n_diff_pfx01 |       9982 |          20 | id                                |
| test          | single_table | PRIMARY      | 2023-10-25 09:56:21 | n_leaf_pages |         62 |        NULL | Number of leaf pages in the index |
| test          | single_table | PRIMARY      | 2023-10-25 09:56:21 | size         |         97 |        NULL | Number of pages in the index      |
| test          | single_table | idx_key1     | 2023-10-25 09:56:21 | n_diff_pfx01 |       9904 |          16 | key1                              |
| test          | single_table | idx_key1     | 2023-10-25 09:56:21 | n_diff_pfx02 |       9904 |          16 | key1,id                           |
| test          | single_table | idx_key1     | 2023-10-25 09:56:21 | n_leaf_pages |         16 |        NULL | Number of leaf pages in the index |
| test          | single_table | idx_key1     | 2023-10-25 09:56:21 | size         |         17 |        NULL | Number of pages in the index      |
| test          | single_table | idx_key2     | 2023-10-25 09:56:21 | n_diff_pfx01 |       9904 |          10 | key2                              |
| test          | single_table | idx_key2     | 2023-10-25 09:56:21 | n_leaf_pages |         10 |        NULL | Number of leaf pages in the index |
| test          | single_table | idx_key2     | 2023-10-25 09:56:21 | size         |         11 |        NULL | Number of pages in the index      |
| test          | single_table | idx_key3     | 2023-10-25 09:56:21 | n_diff_pfx01 |       9904 |          16 | key3                              |
| test          | single_table | idx_key3     | 2023-10-25 09:56:21 | n_diff_pfx02 |       9904 |          16 | key3,id                           |
| test          | single_table | idx_key3     | 2023-10-25 09:56:21 | n_leaf_pages |         16 |        NULL | Number of leaf pages in the index |
| test          | single_table | idx_key3     | 2023-10-25 09:56:21 | size         |         17 |        NULL | Number of pages in the index      |
| test          | single_table | idx_key_part | 2023-10-25 09:56:21 | n_diff_pfx01 |       9905 |          33 | key_part1                         |
| test          | single_table | idx_key_part | 2023-10-25 09:56:21 | n_diff_pfx02 |       9905 |          33 | key_part1,key_part2               |
| test          | single_table | idx_key_part | 2023-10-25 09:56:21 | n_diff_pfx03 |       9905 |          33 | key_part1,key_part2,key_part3     |
| test          | single_table | idx_key_part | 2023-10-25 09:56:21 | n_diff_pfx04 |       9905 |          33 | key_part1,key_part2,key_part3,id  |
| test          | single_table | idx_key_part | 2023-10-25 09:56:21 | n_leaf_pages |         33 |        NULL | Number of leaf pages in the index |
| test          | single_table | idx_key_part | 2023-10-25 09:56:21 | size         |         97 |        NULL | Number of pages in the index      |
+---------------+--------------+--------------+---------------------+--------------+------------+-------------+-----------------------------------+
20 rows in set (0.00 sec)

说明:

  • 先查看 index_name 列,这个列说明该记录是哪个索引的统计信息,从结果中我们可以看出来, PRIMARY 索引(也就是主键)占了3条记录, idx_key1 索引占了4条记录。
  • 针对 index_name 列相同的记录, stat_name 表示针对该索引的统计项名称, stat_value 展示的是该索引在该统计项上的值, stat_description 指的是来描述该统计项的含义的。我们来具体看一下一个索引都有哪些统计项:
    • n_leaf_pages :表示该索引的叶子节点占用多少页面。
    • size :表示该索引共占用多少页面。
    • n_diff_pfxNN :表示对应的索引列不重复的值有多少。其实 NN 可以被替换为 01 、 02 、 03 … 这样的数字。比如对于 idx_key_part 来说:
      • n_diff_pfx01 表示的是统计 key_part1 这单单一个列不重复的值有多少。
      • n_diff_pfx02 表示的是统计 key_part1、key_part2 这两个列组合起来不重复的值有多少。
      • n_diff_pfx03 表示的是统计 key_part1、key_part2、key_part3 这三个列组合起来不重复的值有多少。
      • n_diff_pfx04 表示的是统计 key_part1、key_part2、key_part3、id 这四个列组合起来不重复的值有多少。

注意:对于普通的二级索引,并不能保证它的索引列值是唯一的,比如对于idx_key1来说,key1列就可能有很多值重复的记录。此时只有在索引列上加上主键值才可以区分两条索引列值都一样的二级索引记录。对于主键和唯一二级索引则没有这个问题,
它们本身就可以保证索引列值的不重复,所以也不需要再统计一遍在索引列后加上主键值的不重复值有多少。比如上边的idx_key1有n_diff_pfx01、n_diff_pfx02两个统计项,而idx_key2却只有n_diff_pfx01一个统计项。

  • 在计算某些索引列中包含多少不重复值时,需要对一些叶子节点页面进行采样, size 列就表明了采样的页面数量是多少。

注意:对于有多个列的联合索引来说,采样的页面数量是:innodb_stats_persistent_sample_pages× 索引列的个数。当需要采样的页面数量大于该索引的叶子节点数量的话,就直接采用全表扫描来统计索引列的不重复值数量了。所以大家可以在查询结果中看到不同索引对应的size列的值可能是不同的。

定期更新统计数据

随着我们不断的对表进行增删改操作,表中的数据也一直在变化, innodb_table_stats 和 innodb_index_stats表里的统计数据也会跟着一起更新变化的。在MySQL中提供了如下两种更新统计数据的方式:

  • 开启 innodb_stats_auto_recalc 。
    系统变量 innodb_stats_auto_recalc 决定着服务器是否自动重新计算统计数据,它的默认值是 ON ,也就是该功能默认是开启的。每个表都维护了一个变量,该变量记录着对该表进行增删改的记录条数,如果发生变动的记录数量超过了表大小的 10% ,并且自动重新计算统计数据的功能是打开的,那么服务器会重新进行一次统计数据的计算,并且更新 innodb_table_stats 和 innodb_index_stats 表。不过自动重新计算统计数据的过程是异步发生的,也就是即使表中变动的记录数超过了 10% ,自动重新计算统计数据也不会立即发生,可能会延迟几秒才会进行计算。

说明, InnoDB 默认是以表为单位来收集和存储统计数据的,我们也可以单独为某个表设置是否自动重新计算统计数的属性,设置方式就是在创建或修改表的时候通过指定 STATS_AUTO_RECALC 属性来指明该表的统计数据存储方式:
CREATE TABLE 表名 (…) Engine=InnoDB, STATS_AUTO_RECALC = (1|0);
ALTER TABLE 表名 Engine=InnoDB, STATS_AUTO_RECALC = (1|0);

当 STATS_AUTO_RECALC=1 时,表明我们想让该表自动重新计算统计数据,当 STATS_PERSISTENT=0 时,表明不想让该表自动重新计算统计数据。如果我们在创建表时未指定 STATS_AUTO_RECALC 属性,那默认采用系统变量 innodb_stats_auto_recalc 的值作为该属性的值。

  • 手动调用 ANALYZE TABLE 语句来更新统计信息
    如果 innodb_stats_auto_recalc 系统变量的值为 OFF 的话,我们也可以手动调用 ANALYZE TABLE 语句来重新计算统计数据,比如我们可以这样更新关于single_table 表的统计数据:
mysql> use test;
Database changed
mysql> ANALYZE TABLE single_table;
+-------------------+---------+----------+----------+
| Table             | Op      | Msg_type | Msg_text |
+-------------------+---------+----------+----------+
| test.single_table | analyze | status   | OK       |
+-------------------+---------+----------+----------+
1 row in set (0.04 sec)

mysql> 

注意:执行NAL YZE T ABLE语句会立即重新计算统计数据,也就是这个过程是同步的,在表中索引多或者采样页面特别多时这个过程可能会特别慢,所以最好不要经常运行 ANALYZE TABLE 语句,而且在业务量小的时候再执行。

手动更新 innodb_table_stats 和 innodb_index_stats 表

其实 innodb_table_stats 和 innodb_index_stats 表就相当于一个普通的表一样,我们也能对它们做增删改查操作。这也就意味着我们可以手动更新某个表或者索引的统计数据。比如说我们想把 single_table 表关于行数的统计数据更改一下可以这么做:

  • 步骤一:更新 innodb_table_stats 表。
    UPDATE innodb_table_stats
    SET n_rows = 1
    WHERE table_name = ‘single_table’;
  • 步骤二:让 MySQL 查询优化器重新加载我们更改过的数据。
    更新完 innodb_table_stats 只是单纯的修改了一个表的数据,需要让 MySQL 查询优化器重新加载我们更改过的数据,运行下边的命令就可以了:
    FLUSH TABLE single_table;

之后我们使用 SHOW TABLE STATUS 语句查看表的统计数据时就看到 Rows 行变为了 1 。

基于内存的非永久性统计数据

当我们把系统变量 innodb_stats_persistent 的值设置为 OFF 时,之后创建的表的统计数据默认就都是非永久性的了,或者我们直接在创建表或修改表时设置 STATS_PERSISTENT 属性的值为 0 ,那么该表的统计数据就是非永久性的了。
与永久性的统计数据不同,非永久性的统计数据采样的页面数量是由 innodb_stats_transient_sample_pages 控制的,这个系统变量的默认值是 8 。这个是否使用跟使用的版本有关。

innodb_stats_method的使用

我们知道索引列不重复的值的数量这个统计数据对于 MySQL 查询优化器十分重要,因为通过它可以计算出在索引列中平均一个值重复多少行,它的应用场景主要有两个:

  • 单表查询中单点区间太多,比方说这样:
    SELECT * FROM tbl_name WHERE key IN (‘xx1’, ‘xx2’, …, ‘xxn’);
    当 IN 里的参数数量过多时,采用 index dive 的方式直接访问 B+ 树索引去统计每个单点区间对应的记录的数量就太耗费性能了,所以直接依赖统计数据中的平均一个值重复多少行来计算单点区间对应的记录数量。
  • 连接查询时,如果有涉及两个表的等值匹配连接条件,该连接条件对应的被驱动表中的列又拥有索引时,则
    可以使用 ref 访问方法来对被驱动表进行查询,比方说这样:
    SELECT * FROM t1 JOIN t2 ON t1.column = t2.key WHERE …;
    在真正执行对 t2 表的查询前, t1.comumn 的值是不确定的,所以我们也不能通过 index dive 的方式直接访问 B+ 树索引去统计每个单点区间对应的记录的数量,所以也只能依赖统计数据中的平均一个值重复多少行来计算单点区间对应的记录数量。

在统计索引列不重复的值的数量时,有一个比较烦的问题就是索引列中出现 NULL 值怎么办,比方说某个索引列
的内容是这样:
±-----+
| col |
±-----+
| 1 |
| 2 |
| NULL |
| NULL |
±-----+
此时计算这个 col 列中不重复的值的数量就有下边的分歧:

  • 有的人认为 NULL 值代表一个未确定的值,所以在 MySQL 中任何和 NULL 值做比较的表达式的值都为 NULL ,如下:
mysql> SELECT 1 = NULL;
+----------+
| 1 = NULL |
+----------+
|     NULL |
+----------+
1 row in set (0.00 sec)
mysql> SELECT 1 != NULL;
+-----------+
| 1 != NULL |
+-----------+
|      NULL |
+-----------+
1 row in set (0.00 sec)

mysql> SELECT NULL != NULL;
+--------------+
| NULL != NULL |
+--------------+
|         NULL |
+--------------+
1 row in set (0.03 sec)

mysql> SELECT NULL = NULL;
+-------------+
| NULL = NULL |
+-------------+
|        NULL |
+-------------+
1 row in set (0.00 sec)

所以每一个NULL值都是独一无二的,也就是说统计索引列不重复的值的数量时,应该把NULL值当作一个独立的值,所以col列的不重复的值的数量就是:4(分别是1、2、NULL、NULL这四个值)。

  • 有的人认为其实 NULL 值在业务上就是代表没有,所有的 NULL 值代表的意义是一样的,所以 col 列不重复的值的数量就是: 3 (分别是1、2、NULL这三个值)。
  • 有的人认为这 NULL 完全没有意义,所以在统计索引列不重复的值的数量时压根儿不能把它们算进来,所以 col 列不重复的值的数量就是: 2 (分别是1、2这两个值)。

在Mysql中提供了一个名为 innodb_stats_method 的系统变量,相当于在计算某个索引列不重复值的数量时如何对待 NULL 值这个决定权给了用户,这个系统变量有三个候选值:

  • nulls_equal :认为所有 NULL 值都是相等的。这个值也是 innodb_stats_method 的默认值。
    如果某个索引列中 NULL 值特别多的话,这种统计方式会让优化器认为某个列中平均一个值重复次数特别多,所以倾向于不使用索引进行访问。
  • nulls_unequal :认为所有 NULL 值都是不相等的。
    如果某个索引列中 NULL 值特别多的话,这种统计方式会让优化器认为某个列中平均一个值重复次数特别少,所以倾向于使用索引进行访问。
  • nulls_ignored :直接把 NULL 值忽略掉。

当选定了 innodb_stats_method 值之后,优化器即使选择了不是最优的执行计划,那就是自己设置的问题。所以对于用户来说,最好不在索引列中存放NULL值才是明智之举。

总结

  • innoDB 以表为单位来收集统计数据,这些统计数据可以是基于磁盘的永久性统计数据,也可以是基于内存的非永久性统计数据。
  • innodb_stats_persistent 控制着使用永久性统计数据还是非永久性统计数据;
  • innodb_stats_persistent_sample_pages 控制着永久性统计数据的采样页面数量;
  • innodb_stats_transient_sample_pages 控制着非永久性统计数据的采样页面数量;
  • innodb_stats_auto_recalc 控制着是否自动重新计算统计数据。
  • 我们可以针对某个具体的表,在创建和修改表时通过指定 STATS_PERSISTENT 、 STATS_AUTO_RECALC 、
    STATS_SAMPLE_PAGES 的值来控制相关统计数据属性。
  • innodb_stats_method 决定着在统计某个索引列不重复值的数量时如何对待 NULL 值。

更多关于mysql的知识分享,请前往博客主页。编写过程中,难免出现差错,敬请指出

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

深圳锐科达SIP矿用电话模块SV-2801VP

深圳锐科达SIP矿用电话模块SV-2801VP 一、简介 SV-2800VP系列模块是我司设计研发的一款用于井下的矿用IP音频传输模块,可用此模块打造一套低延迟、高效率、高灵活和多扩展的IP矿用广播对讲系统,亦可对传统煤矿电话系统加装此模块,进行智能化…

在Vue3中使用vue-qrcode库实现二维码生成

本文主要介绍在Vue3中使用qrcode库实现二维码生成的方法。 目录 一、基础用法实现vue-qrcode库的参数介绍 在Vue3中实现二维码生成需要使用第三方库来处理生成二维码的逻辑。常用的库有 qrcode和 vue-qrcode。 一、基础用法实现 在Vue3中使用vue-qrcode库实现二维码生成的方…

记录一次云主机故障排查

云上某云主机,在安全组test-a中,同一安全组下还有另外两台主机。 从本地可以ping 通另外两台主机的公网地址。但是不能ping通这个主机的公网地址。 与是重启主机,发现问题依然存在。依然是不能ping 通,主机上部署的业务也不能访…

LabVIEW在齿轮箱故障诊断中的应用

LabVIEW在齿轮箱故障诊断中的应用 在现代机械工业中,齿轮箱作为重要的传动设备,其性能稳定性对整体机械系统的运行至关重要。故障的及时诊断和处理不仅保障了设备的稳定运行,还减少了维护成本。利用LabVIEW强大数据处理和仿真能力&#xff0…

新版IDEA中Git的使用(二)

说明:前面介绍了在新版IDEA中Git的基本操作,本文介绍关于分支合并、拉取等操作; 例如,现在有一个项目,分支如下: main:主分支; dev:开发分支; test&#x…

Springboot整合MVC进阶篇

一、概述 1.1SpringBoot整合SpringMVC配置 SpringBoot对SpringMVC的配置主要包括以下几个方面: 自动配置:SpringBoot会自动配置一个嵌入式的Servlet容器(如Tomcat),并为我们提供默认的SpringMVC配置。这样我们无需手动…

【Java、Python】获取电脑当前网络IP进行位置获取(附源码)

我相信看到这篇博客的时候心里肯定是想解决自己的一个问题的,而这篇博客我就以简单快速的方式解决这些烦恼! 一、获取当前IP 在Java中自带了一些自己的流对象来获取当前的IP地址,不多说我们直接上代码。 //获取当前网络ip地址 ipAddress Ine…

在k8s中将gitlab-runner的运行pod调度到指定节点

本篇和前面的 基于helm的方式在k8s集群中部署gitlab 具有很强的关联性,因此如果有不明白的地方可以查看往期分享: 基于helm的方式在k8s集群中部署gitlab - 部署基于helm的方式在k8s集群中部署gitlab - 备份恢复基于helm的方式在k8s集群中部署gitlab - 升…

论文阅读——X-Decoder

Generalized Decoding for Pixel, Image, and Language Towards a Generalized Multi-Modal Foundation Model 1、概述 X-Decoder没有为视觉和VL任务开发统一的接口,而是建立了一个通用的解码范式,该范式可以通过采用共同的(例如语义&#…

实战:朴素贝叶斯文本分类器搭建与性能评估

💗💗💗欢迎来到我的博客,你将找到有关如何使用技术解决问题的文章,也会找到某个技术的学习路线。无论你是何种职业,我都希望我的博客对你有所帮助。最后不要忘记订阅我的博客以获取最新文章,也欢…

生成超清分辨率视频,南洋理工开源Upscale-A-Video

大模型在生成高质量图像方面表现出色,但在生成视频任务中,经常会面临视频不连贯、图像模糊、掉帧等问题。 这主要是因为生成式抽样过程中的随机性,会在视频序列中引入无法预测的帧跳动。同时现有方法仅考虑了局部视频片段的时空一致性,无法保证整个长视频的整体连贯…

基于电商场景的高并发RocketMQ实战-Broker写入读取流程性能优化总结、Broker基于Pull模式的主从复制原理

🌈🌈🌈🌈🌈🌈🌈🌈 【11来了】文章导读地址:点击查看文章导读! 🍁🍁🍁🍁🍁🍁&#x1f3…

关于OpenCV中 CV_Assert() 的使用引起程序中止/崩溃问题

CV_Assert() 的作用是:若括号中的表达式值为 false ,则返回一个错误信息,并终止程序执行。 但是 CV_Assert() 与 assert 不同,CV_Assert() 会通过异常抛出,所以如果使用 CV_Assert(),可以通过捕获异常而不是…

三列布局 css

实现如下图的三列布局: .box {width:1400px;margin:0 auto;padding-bottom:40px;> .left {float:left;width:180px;margin-top:100px;text-align:center;}> .center {float:left;margin-top:100px;margin-left:130px;item-box {float:left;text-align:left;…

oom问题

问题描述 虚拟机集群节点上pod报oom,最后pod被驱逐,主节点上查看kubectl top node的mem使用率很高,重启系统后,mem会降下来,但还会慢慢增长。 node节点上查看 /sys/fs/cgroup/memory/memory.usage_in_bytes内存使用超…

Spring Boot简单多线程定时任务实现 | @Async | @Scheduled

Spring Boot简单多线程定时任务实现 实现步骤 1 创建一个Spring Boot项目 2 定义定时任务: package com.jmd.timertasktest.task;import org.springframework.context.annotation.Configuration; import org.springframework.scheduling.annotation.Async; impor…

GenerateBlocks Pro插件 构建更好的WordPress网站

GenerateBlocks Pro插件 构建更好的WordPress网站 GenerateBlocks Pro插件是一个 WordPress 插件,几乎可以完成任何事情,可让您创建轻量级和多功能的网站。由与流行且快速的 GeneratePress 主题相同的创作者构建,该插件不负众望。使用 Genera…

H266/VVC帧间预测编码技术概述

帧间预测编码简述 帧间预测利用视频时间域的相关性,使用邻近已编码图像像素值预测当前图像的像素值,能有效去除视频时域冗余。 目前主要的视频编码标准中,帧间预测都采用基于块的运动补偿技术,不同的编码标准有不同的分块方式。 …

智能优化算法应用:基于驾驶训练算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用:基于驾驶训练算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用:基于驾驶训练算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.驾驶训练算法4.实验参数设定5.算法结果6.…

Android笔记(二十一):Room组件实现Android应用的持久化处理

一、Room组件概述 Room是Android JetPack架构组件之一,是一个持久处理的库。Room提供了在SQLite数据库上提供抽象层,使之实现数据访问。 (1)实体类(Entity):映射并封装了数据库对应的数据表中…