0

0

mysql执行过程中如何更新索引统计信息_mysql统计信息刷新触发

P粉602998670

P粉602998670

发布时间:2026-03-15 16:04:02

|

649人浏览过

|

来源于php中文网

原创

ANALYZE TABLE 会在表首次被打开、显式执行该命令、或 innodb_stats_auto_recalc=ON 且变更行数超预估行数10%并含二级索引时自动触发;它通过采样20个数据页估算统计信息,非全表扫描,结果存在误差。

mysql执行过程中如何更新索引统计信息_mysql统计信息刷新触发

什么时候 ANALYZE TABLE 会自动触发?

MySQL 不会在每次 INSERT/UPDATE 后自动更新索引统计信息,只有少数明确场景下才会触发:表首次被打开、执行 ANALYZE TABLE、或开启 innodb_stats_auto_recalc=ON 且表变更行数超 10%(默认阈值)且有二级索引时。注意:这个“10%”是基于 INFORMATION_SCHEMA.INNODB_TABLESTATS 中的预估行数,不是实时精确值。

常见误解是“只要数据变了,执行计划就立刻变”,其实不会。比如批量导入 5 万行后直接 EXPLAIN,优化器仍可能沿用旧统计信息,导致走错索引。

  • innodb_stats_auto_recalc 默认为 ON,但只对 CREATE TABLE ... STATS_AUTO_RECALC=1 创建的表生效;已有表需手动 ALTER TABLE t STATS_AUTO_RECALC=1
  • 临时表、内存表、分区表的子分区不参与自动统计更新
  • 如果 innodb_stats_persistent=OFF(老版本默认),统计信息只存在内存中,重启后丢失,必须手动 ANALYZE

ANALYZE TABLE 执行时会发生什么?

它不是“扫描全表”,而是采样估算:InnoDB 默认采样 20 个数据页(由 innodb_stats_sample_pages 控制),每页取部分记录构建列值分布和基数(cardinality)。这意味着结果本身就有误差,尤其当数据倾斜严重(如某字段 90% 是 'active')时,cardinality 可能严重失真。

执行期间会加 MDL 共享锁(非阻塞 DML),但大表采样可能持续数秒——这不是 I/O 瓶颈,而是采样逻辑本身需要遍历 B+ 树叶子页链表。

  • 采样页数可临时调高:SET GLOBAL innodb_stats_sample_pages = 100,但仅对后续 ANALYZE 生效,且过高会延长执行时间
  • ANALYZE TABLE 不阻塞 SELECT/INSERT,但会短暂阻塞其他 ANALYZEALTER
  • 对 MyISAM 表是直接计算精确值,行为完全不同,别混用

为什么 EXPLAIN 显示的 rows 和实际相差巨大?

因为 rows 来自统计信息里的 cardinality 估算,而该值依赖两个关键前提:数据均匀分布、采样足够代表整体。现实里这两点常不成立。例如一个时间戳字段,新数据集中在最近 1 小时,但采样页随机落在历史冷区,就会低估范围查询的匹配行数。

In3D
In3D

把真人变成化身,创建逼真且可自定义的虚拟角色

下载

另一个典型坑是:使用了前缀索引(如 INDEX (name(10))),统计信息只基于前 10 字符计算 cardinality,但 WHERE name = 'JohnDoe123' 实际匹配靠的是完整值——此时优化器误判选择性,可能放弃索引。

  • 检查当前统计是否陈旧:SELECT * FROM INFORMATION_SCHEMA.STATISTICS WHERE TABLE_NAME = 't' AND TABLE_SCHEMA = 'db',看 INDEX_COMMENT 是否含 stats_cached
  • 强制刷新单个索引:ANALYZE TABLE t UPDATE HISTOGRAM ON col1, col2(MySQL 8.0+),比全表 ANALYZE 更轻量
  • 避免在 WHERE 中对索引列用函数:WHERE YEAR(created_at) = 2024 会让统计信息完全失效

生产环境怎么安全地更新统计信息?

不能靠“定时 ANALYZE TABLE”,因为大表可能卡住备份或慢查询日志归档。更稳妥的方式是结合数据变更特征做条件触发:比如 ETL 任务结束后、每日凌晨低峰期、或监控到某表 INFORMATION_SCHEMA.INNODB_TABLESTATSMODIFIED_COUNTER 突增超过阈值时再执行。

还要注意权限:执行 ANALYZE TABLE 需要 PROCESS 权限(某些托管 MySQL 如 RDS 默认关闭),且对系统库表无效。

  • 脚本化示例:mysql -e "SELECT CONCAT('ANALYZE TABLE ', TABLE_SCHEMA, '.', TABLE_NAME, ';') FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = 'prod' AND TABLE_ROWS > 100000;" | mysql
  • 跳过临时表和视图:AND TABLE_TYPE = 'BASE TABLE'
  • 如果用了 pt-online-schema-change,它默认会自动 ANALYZE 新表,无需额外操作

统计信息不是越新越好,频繁更新反而让执行计划抖动。关键是理解它怎么影响优化器决策,而不是把它当成必须“保持最新”的配置项。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
mysql修改数据表名
mysql修改数据表名

MySQL修改数据表:1、首先查看数据库中所有的表,代码为:‘SHOW TABLES;’;2、修改表名,代码为:‘ALTER TABLE 旧表名 RENAME [TO] 新表名;’。php中文网还提供MySQL的相关下载、相关课程等内容,供大家免费下载使用。

686

2023.06.20

MySQL创建存储过程
MySQL创建存储过程

存储程序可以分为存储过程和函数,MySQL中创建存储过程和函数使用的语句分别为CREATE PROCEDURE和CREATE FUNCTION。使用CALL语句调用存储过程智能用输出变量返回值。函数可以从语句外调用(通过引用函数名),也能返回标量值。存储过程也可以调用其他存储过程。php中文网还提供MySQL创建存储过程的相关下载、相关课程等内容,供大家免费下载使用。

534

2023.06.21

mongodb和mysql的区别
mongodb和mysql的区别

mongodb和mysql的区别:1、数据模型;2、查询语言;3、扩展性和性能;4、可靠性。本专题为大家提供mongodb和mysql的区别的相关的文章、下载、课程内容,供大家免费下载体验。

287

2023.07.18

mysql密码忘了怎么查看
mysql密码忘了怎么查看

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql密码忘了怎么办呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

520

2023.07.19

mysql创建数据库
mysql创建数据库

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql怎么创建数据库呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

267

2023.07.25

mysql默认事务隔离级别
mysql默认事务隔离级别

MySQL是一种广泛使用的关系型数据库管理系统,它支持事务处理。事务是一组数据库操作,它们作为一个逻辑单元被一起执行。为了保证事务的一致性和隔离性,MySQL提供了不同的事务隔离级别。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

392

2023.08.08

sqlserver和mysql区别
sqlserver和mysql区别

SQL Server和MySQL是两种广泛使用的关系型数据库管理系统。它们具有相似的功能和用途,但在某些方面存在一些显著的区别。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

542

2023.08.11

mysql忘记密码
mysql忘记密码

MySQL是一种关系型数据库管理系统,关系数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。那么忘记mysql密码我们该怎么解决呢?php中文网给大家带来了相关的教程以及其他关于mysql的文章,欢迎大家前来学习阅读。

668

2023.08.14

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

69

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 2.6万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 850人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号