49人参与 • 2025-04-21 • Mysql
在使用mysql进行模糊查询时,like语句的性能可能会受到较大影响,尤其是在数据量较大的情况下。
但本质上,用like进行模糊查询,只有以下三种情况:
前缀匹配(如like 'abc%')可以使用b-tree索引,因此性能较好。确保在相关列上创建索引
示例:
-- 创建表 create table users ( id int primary key auto_increment, username varchar(255) not null ); -- 插入数据 insert into users (username) values ('john_doe'), ('jane_doe'), ('alice'), ('bob'), ('john_smith'); -- 创建索引 create index idx_username on users(username); -- 前缀匹配查询 explain select * from users where username like 'john%';
后缀匹配(如like '%abc'),无法直接使用b-tree索引,可以通过反转字符串并创建索引来优化
示例:
-- 添加反转列 alter table users add column reversed_username varchar(255); -- 更新反转列数据 update users set reversed_username = reverse(username); -- reverse('hello') 的结果是 'olleh' -- 创建反转列索引 create index idx_reversed_username on users(reversed_username); -- 后缀匹配查询(转换为前缀匹配) explain select * from users where reversed_username like reverse('doe') + '%';
中间匹配(如like '%abc%')无法使用b-tree索引。可以考虑使用全文索引或外部搜索引擎
示例(使用全文索引)
-- 创建全文索引 create fulltext index idx_username_fulltext on users(username); -- 全文索引查询 explain select * from users where match(username) against('doe');
如果查询只需要返回索引列,可以使用覆盖索引(covering index),避免回表操作
示例:
-- 创建覆盖索引 create index idx_username_covering on users(username, id); -- 覆盖索引查询 explain select username from users where username like 'john%';
通过其他条件缩小查询范围,减少模糊查询的数据量
示例:
-- 假设有一个注册时间列 alter table users add column registered_at datetime; -- 插入数据 update users set registered_at = now() - interval floor(rand() * 365) day; -- 缩小查询范围 explain select * from users where registered_at > '2023-01-01' and username like 'john%';
尽量避免在like语句中使用通配符开头(如%abc),因为这种查询无法使用索引
示例:
-- 不推荐的查询 explain select * from users where username like '%doe'; -- 优化后的查询(使用全文索引) explain select * from users where match(username) against('doe');
type
列会显示all
,表示全表扫描。type
列会显示fulltext
,表示使用了全文索引。对于复杂的模糊查询需求,尤其是大数据量场景,可以使用外部搜索引擎(如elatsticsearch)
示例
如果数据量非常大,可以使用分区表(partitioning),来较少每次查询需要扫描的数据量
示例:
-- 创建分区表 create table users_partitioned ( id int primary key auto_increment, username varchar(255) not null, registered_at datetime ) partition by range (year(registered_at)) ( partition p0 values less than (2020), partition p1 values less than (2021), partition p2 values less than (2022), partition p3 values less than (2023), partition p4 values less than maxvalue ); -- 插入数据 insert into users_partitioned (username, registered_at) select username, registered_at from users; -- 分区表查询 explain select * from users_partitioned where registered_at > '2023-01-01' and username like 'john%';
explain
结果中的partitions
列会显示查询涉及的分区,表明查询只扫描了部分数据。如果模糊查询的结果不经常变化,可以将查询结果缓存起来,减少数据库的查询压力
示例:
通过以上方法,可以显著优化mysql中like
模糊查询的性能。根据具体的业务需求和数据特点,选择合适的优化策略:
注:了解mysql-match ... against工具参考mysql-match ... against工具
到此这篇关于mysql--索引的优化--like模糊查询的文章就介绍到这了,更多相关mysql like模糊查询内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!
您想发表意见!!点此发布评论
版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。
发表评论