加索引后查询仍慢的主因是索引未覆盖、顺序不当或隐式类型转换致失效;高并发下卡顿多由锁争用、元数据锁等待或主从延迟引发。

为什么加索引后查询还是慢?
高并发下即使有索引,SELECT 仍可能卡住,常见原因是索引未覆盖查询字段、索引列顺序不合理,或存在隐式类型转换导致索引失效。比如 WHERE user_id = '123'(user_id 是 INT),MySQL 会强制转为数字再比对,但某些版本下会放弃使用索引。
实操建议:
- 用
EXPLAIN FORMAT=TREE查看执行计划,确认是否走了预期索引,尤其注意filtered值是否远低于 100(说明实际扫描行数远超预估) - 联合索引要遵循「最左前缀」,且把高频过滤、区分度高的列放前面;排序字段若在
ORDER BY中出现,尽量包含在索引末尾(避免Using filesort) - 避免在索引列上做函数操作,如
WHERE DATE(create_time) = '2024-01-01'→ 改成WHERE create_time >= '2024-01-01' AND create_time
连接池配置不当会直接拖垮QPS
应用层连接池(如 HikariCP、Druid)若最大连接数设得过高,MySQL 的 max_connections 很快被耗尽,新请求排队等待,表现为连接超时或 Too many connections 错误;设得太低又会造成线程饥饿,吞吐上不去。
实操建议:
- 先查 MySQL 实际峰值连接数:
SHOW STATUS LIKE 'Threads_connected';,结合监控观察 95 分位值,max_connections建议设为该值的 1.5–2 倍 - HikariCP 中
maximumPoolSize不宜超过 20–30(除非 IO 密集型且数据库硬件极强),并开启connection-timeout(推荐 3000ms)防止阻塞扩散 - 务必关闭自动提交(
autoCommit=false),显式控制事务边界,避免长事务占用连接
读写分离没生效?可能是事务里混了写操作
很多业务以为只要用了主从架构,查询就会自动走从库——其实不然。一旦当前连接处于事务中(哪怕只执行了一条 SELECT),后续所有语句(包括读)都会路由到主库,否则无法保证一致性。
系统功能强大、操作便捷并具有高度延续开发的内容与知识管理系统,并可集合系统强大的新闻、产品、下载、投票、人才、留言、在线订购、搜索引擎优化、等功能模块,为企业部门提供一个简单、易用、开放、可扩展的企业信息门户平台或电子商务运行平台。开发人员为脆弱页面专门设计了防刷新系统,自动阻止恶意访问和攻击;安全检查应用于每一处代码中,每个提交到系统查询语句中的变量都经过过滤,可自动屏蔽恶意攻击代码,从而全面防
实操建议:
- 检查应用代码:事务方法内是否夹杂了
INSERT/UPDATE/DELETE,哪怕只是日志表写入,也会让整个事务绑定主库 - 用
SELECT @@innodb_read_only;在从库上确认只读状态是否开启(应返回 1),同时确认主从延迟Seconds_Behind_Master是否稳定在 100ms 内 - 对强一致性要求不高的场景(如商品详情页浏览),可手动指定从库数据源,绕过框架默认路由逻辑
limit深分页为什么越翻越卡?
SELECT * FROM order WHERE status = 1 ORDER BY id LIMIT 10000, 20 这类语句在高并发下极易成为瓶颈:MySQL 必须先扫描前 10020 行,再丢弃前 10000 行,IO 和 CPU 开销随偏移量线性增长。
实操建议:
- 改用游标分页(cursor-based pagination):记录上一页最后一条的
id,下一页查WHERE id > 12345 AND status = 1 ORDER BY id LIMIT 20 - 对必须支持跳页的后台系统,可建冗余字段(如
list_order)并维护有序序列,用等值查询替代LIMIT offset - 禁止前端传入超大
offset(如 > 1000),API 层直接拦截并返回 400
真正卡住高并发的,往往不是单条 SQL 多慢,而是连接争抢、锁等待、主从延迟放大后的连锁反应。调优时优先看 SHOW PROCESSLIST 里有没有大量 Waiting for table metadata lock 或 Sending data 状态,这些信号比慢日志更早暴露系统瓶颈。










