【MySQL-索引调优】08:模糊查询与索引

PS:给order表新增remark索引

sql 复制代码
CREATE INDEX idx_remark ON orders(remark);

现在orders表中存在idx_user_id(user_id),idx_user_status(user_id, status),idx_remark(remark)三个索引

8-1.后置%

运行:

sql 复制代码
EXPLAIN SELECT * FROM orders WHERE remark LIKE '71ae%';

结果为:

id select_type table partitions type possible_keys key key_len ref rows filtered Extra
1 SIMPLE orders range idx_remark idx_remark 1023 207 100 Using index condition

分析:

  • type = range :表示这是一个范围扫描。因为 LIKE '71ae%' 在索引上相当于一个范围条件:大于等于 '71ae' 且小于 '71af'(实际上是以 '71ae' 开头的所有字符串)。所以MySQL会用索引做范围查找
  • possible_keys = idx_remark :数据库识别到可以用 remark 的索引
  • key = idx_remark:实际使用的就是你刚建的索引
  • rows = 207:预估只需要扫描207行,全表扫描已经被优化掉了
  • filtered = 100:表示这207行全部符合条件
  • Extra = Using index condition :说明使用了 Index Condition Pushdown (ICP) 优化:过滤条件直接在索引层完成,而不是先取出数据再过滤,这进一步减少了回表的开销

8-2.中间%

运行:

sql 复制代码
EXPLAIN SELECT * FROM orders WHERE remark LIKE '%71ae%';

结果为:

id select_type table partitions type possible_keys key key_len ref rows filtered Extra
1 SIMPLE orders ALL 99520 11.11 Using where

分析:

LIKE '%71ae%',索引无法使用。在 MySQL 中,B+Tree 索引必须从左侧开始匹配

因为索引 idx_remark 的结构类似:

复制代码
71aa....
71ab....
71ac....
71ae....
71af....

如果查询:

复制代码
LIKE '71ae%'

数据库可以直接定位:

复制代码
71ae 开头的范围

但如果查询:

复制代码
LIKE '%71ae%'

数据库不知道

复制代码
71ae在字符串什么位置

总结:

  • LIKE 'abc%' → 可以使用索引
  • LIKE '%abc' → 无法使用索引
  • LIKE '%abc%' → 无法使用索引

8-3.真实系统如何解决%xxx%查询

  1. 方案1:全文索引

MySQL提供:

复制代码
FULLTEXT INDEX

适合:文章、评论、文本

  1. 方案2:搜索引擎

例如:Elasticsearch

适合:模糊搜索、关键词搜索

  1. 方案3:前缀搜索

系统设计时只允许:

复制代码
prefix search

例如:订单号、用户ID、编码

相关推荐
百结2142 小时前
Nginx性能优化与监控实战
java·nginx·性能优化
项目工程打工马2 小时前
Ubuntu 上 MySQL 详细安装指南
mysql·ubuntu·adb
数据知道2 小时前
《深入掌握MongoDB数据库》 - 专栏介绍和目录
网络·数据库·mongodb
moxiaoran57532 小时前
使用ShardingSphere实现MySQL读写分离(一)
数据库·mysql
J超会运2 小时前
OpenEuler系统Nginx性能优化全攻略
运维·nginx·性能优化
CoovallyAIHub2 小时前
BMW GenAI4Q:每57秒下线一辆车,AI如何为每辆车定制专属质检清单
数据库·算法·架构
wang2455981992 小时前
Redis基础——1、Linux下安装Redis(超详细)
linux·数据库·redis
oscar9992 小时前
Memurai:Redis官方认可的Windows原生解决方案
数据库·windows·redis
A10169330712 小时前
redis的启动方式
数据库·redis·bootstrap