加索引后select仍慢,主因是并发下索引引发锁争用、内存压力及配置失衡。需用explain确认走索引、优化联合索引顺序、调优buffer_pool_size、避免for update死锁,并禁用query cache。

为什么加了索引,SELECT 还是慢?
索引不是万能的,尤其在并发查询下,它可能变成锁争用源或内存压力点。常见现象是:单条 SELECT 很快,但 50 个并发一跑,响应时间陡增、CPU 拉满、甚至出现 Lock wait timeout exceeded。
- 确认是否真走索引:用
EXPLAIN看type字段,避免ALL或index全扫描;注意key_len是否符合预期(比如只用了联合索引前两列,第三列条件失效) - 联合索引顺序必须匹配
WHERE+ORDER BY的最左前缀,否则排序阶段无法复用索引,触发Using filesort - 高并发读场景下,
READ COMMITTED隔离级别比REPEATABLE READ更轻量,后者在 MySQL 中默认启用间隙锁,容易导致行锁升级为范围锁
innodb_buffer_pool_size 设多大才不拖后腿?
这个参数决定 InnoDB 能缓存多少数据页。设小了,每次查询都去磁盘捞;设大了,又可能挤占系统内存,触发 swap,反而更卡。
- 一般设为物理内存的 50%–75%,但必须留足空间给 OS 和其他进程(比如 PHP-FPM、Redis);线上曾见设到 90%,结果系统频繁 OOM killer 杀掉 mysqld
- 动态调整需重启(MySQL 5.7+ 支持在线调大,但不能缩小),上线前务必压测验证;可用
SHOW ENGINE INNODB STATUS查看Buffer pool hit rate,低于 99% 就该警惕 - 注意监控
Innodb_buffer_pool_wait_free:非零说明 buffer pool 太小,后台刷脏页跟不上分配速度
并发 SELECT ... FOR UPDATE 怎么避免死锁?
这类语句本质是写锁,即使只读也要加行级锁,高并发时极易相互等待形成环路。典型错误是事务里先查再更新,且顺序不一致。
集企业自助建站、网络营销、商品推广于一体的系统 功能说明: 1、系统采用Microsoft SQL Server大型数据库支持,查询数据库用的全是存储过程,速度和性能极好。开发环境是vs.net,采用4层结构,具有很好的可维护性和可扩冲性。 2、用户注册和登陆 未注册用户只具备浏览商品、新闻和留言功能;要采购商品,需接受服务协议并填写相关注册信息成为正式用户后方可进行,以尽可能减少和避免无效
- 尽量用主键或唯一索引做
WHERE条件,避免锁住非目标行(比如用普通索引会锁索引树+对应主键行,范围更大) - 所有业务逻辑中,对同一张表的加锁操作,按固定字段顺序(如始终按
id ASC)获取锁,不要一会儿id > 100,一会儿name LIKE 'a%' - 减少事务内操作:把校验、计算等非 DB 逻辑提到事务外;避免在事务里调外部 API 或 sleep
要不要开 query_cache_type=1?
MySQL 5.7 默认开启但已标记为 deprecated,8.0 直接移除。在并发查询场景下,它反而是性能杀手。
- 查询缓存对每个
SELECT都要加全局锁校验 cache key,QPS 上千时锁竞争严重,实测反而比关闭慢 2–3 倍 - 只要表有任一写入(哪怕
INSERT一行),整个表相关 cache 全部失效,高写入表下命中率趋近于 0 - 替代方案更实在:应用层用 Redis 缓存聚合结果;或用
SELECT SQL_NO_CACHE ...显式绕过(仅调试用)
真正卡住并发查询的,往往不是某条 SQL 写得差,而是多个看似合理的配置叠加后,在连接数、锁粒度、内存分配上悄悄失衡。调优时别只盯着慢日志,得看 SHOW PROCESSLIST 里一堆 Locked 状态,或者 information_schema.INNODB_TRX 里长事务没提交。










