为什么加索引后查询还是慢?
高并发下即使有索引,
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状态,这些信号比慢日志更早暴露系统瓶颈。
