贝利信息

mysql在高并发场景中的索引优化策略

日期:2026-01-09 00:00 / 作者:P粉602998670
根本原因是非唯一二级索引导致间隙锁或临键锁,即使有索引也会锁住大范围;应优先用UNIQUE索引、避免高频字段单独建索引、用联合索引优化,并确保INSERT...ON DUPLICATE KEY UPDATE仅依赖单一唯一索引。

为什么 SELECT ... FOR UPDATE 会卡住,而加了索引也不行?

根本原因不是没加索引,而是加了「非唯一二级索引」却没覆盖查询条件,导致 MySQL 退化为间隙锁(Gap Lock)或临键锁(Next-Key Lock),锁住一大片范围。比如 WHERE status = 1,即使 status 有索引,若该值重复率高,InnoDB 仍可能锁住多个索引项及其间隙。

INSERT ... ON DUPLICATE KEY UPDATE 的索引依赖和死锁风险

这个语句本质是先按唯一约束(主键或 UNIQUE 索引)查找,再决定插入或更新。如果唯一约束不明确、或存在多个 UNIQUE 索引,MySQL 可能加锁顺序不一致,引发死锁。

联合索引的最左匹配失效:为什么 WHERE a = ? AND c = ? 没走索引?

当联合索引是 (a, b, c),而查询跳过中间列 b,MySQL 无法使用 c 部分做索引查找,只能用到 ac 变成回表后过滤。

CREATE INDEX idx_abc ON orders (user_id, status, created_at);

高并发写入时,自增主键为何比 UUID 更稳?

UUID(尤其无序版本如 UUID_SHORT() 或字符串 UUID)会导致聚簇索引频繁页分裂,B+ 树节点反复重排,产生大量磁盘随机写和锁竞争。而 BIGINT AUTO_INCREMENT 是严格递增的,新记录总追加到 B+ 树最右叶子页,写放大最小。

索引不是越多越好,真正关键的是让每条慢查背后的 WHEREORDER BYGROUP BY 都落在一个高效索引的最左前缀上;而高并发下,锁粒度、写入顺序、唯一性保障,往往比“能不能查得快”更致命。