绝大多数等值查询下唯一索引和普通索引性能几乎无差别,核心区别在于唯一索引由数据库强制校验重复值以保障数据一致性,而普通索引仅加速查询且允许重复插入。
直接说结论:**绝大多数等值查询(WHERE column = ?)下,唯一索引和普通索引的性能几乎没差别**。B+树定位到数据页后,InnoDB 都是把整
页读进内存再二分查找;唯一索引多做的只是「找到第一个就停」,普通索引要多判断一次下一行是否还匹配——这个开销在 16KB 数据页里基本可忽略。
但关键不在快不快,而在「数据库是否帮你拦住脏数据」:
INDEX:只加速查询,允许重复值插入,哪怕你重复插 100 次手机号 '13800138000',MySQL 也照收不误UNIQUE:一旦定义,MySQL 会在 INSERT 或 UPDATE 时自动校验——发现重复就报错 ERROR 1062 (23000): Duplicate entry 'xxx' for key 'idx_phone'
UNIQUE 允许 NULL,且多个 NULL 不算重复(这是 SQL 标准行为)典型场景就是业务上天然要求字段全局唯一,且你不想靠应用层反复查库再插入(容易并发冲突):
id_card、email、username
out_trade_no(外部交易号)sn(序列号)UNIQUE(user_id, DATE(created_at))
别图省事只加普通索引再靠代码 try-catch 插入失败——这会把数据一致性压力全推给应用,且并发下极易漏判。
常见错误不是语法报错,而是「你以为建了唯一索引,其实建成了普通索引」:
CREATE INDEX idx_email ON users(email); → 普通索引,无唯一约束CREATE UNIQUE INDEX idx_email ON users(email); → 真正生效的唯一索引ALTER TABLE users ADD UNIQUE(email); → 这条语句会失败(如果已有重复值),但很多人没检查返回值,误以为成功SHOW INDEX FROM users WHERE Key_name = 'idx_email';看
Non_unique 列:0 = 唯一索引,1 = 普通索引唯一索引确实需要额外做「是否存在」校验,但实际影响有限:
0.1ms 级别所以别为了“怕慢”放弃唯一性保障;先确保字段上有对应索引,再谈优化。