索引选择性低(如性别字段)易致全表扫描,因优化器认为回表开销大于遍历全表;选择性=COUNT(DISTINCT col)/COUNT(*),低于0.05~0.1时B+树索引常失效;需结合最左前缀、NULL处理、覆盖索引及EXPLAIN综合判断。
当 WHERE 条件列的选择性很低(比如性别字段只有 '男'/'女' 两种值),MySQL 优化器很可能放弃使用该索引,直接走全表扫描。因为回表或扫描索引本身开销可能比遍历整张表还大。
COUNT(DISTINCT column) / COUNT(*),越接近 1 越好(如用户邮箱)EXPLAIN 查看 type 字段:若为 ALL 或 index,说明没走有效索引MySQL 只能利用联合索引的连续前缀部分。即使 (a, b, c) 中 a 选择性差,但 (a, b) 组合选择性高,查询 WHERE a = ? AND b = ? 仍可高效命中索引。
(status, user_id) 中 status 只有 3 个值)>, BETWEEN)字段必须放最右,否则右侧字段无法被索引使用SHOW INDEX FROM table_name 确认索引定义顺序,别只看 CREATE INDEX 语句写法MySQL 的 ANALYZE TABLE 在统计列基数(Cardinality)时,对含大量 NULL 的列可能严重低估实际区分度,导致优化器误判索引价值。
SELECT COUNT(*), COUNT(column), COUNT(DISTINCT column) FROM table 对比真实选择性NULL 且占比高,考虑改用默认值(如 'unknown')替代,或在查询中显式排除:WHERE column IS NOT NULL AND 
column = ?
innodb_stats_persistent = ON + 定期 ANALYZE TABLE 可缓解统计偏差,但无法根治 NULL 干扰即使筛选字段选择性低,只要查询所需所有字段都在索引中(即覆盖索引),MySQL 就无需回表,此时哪怕走 index 类型扫描,性能也远好于全表扫描。
(status, create_time, id) 联合索引,查询 SELECT id FROM t WHERE status = 'draft' 就是覆盖扫描SELECT * 几乎不可能被覆盖,务必明确列出需要的字段EXPLAIN 检查 Extra 列是否含 Using index,这是覆盖索引生效的关键标志EXPLAIN SELECT id, create_time FROM orders WHERE status = 'shipped';索引选择性不是独立指标,它和查询模式、字段顺序、NULL 分布、是否覆盖紧密耦合。单独优化某列的选择性没意义,得结合
EXPLAIN 输出和实际慢查询场景一起看。