it编程 > 数据库 > Mysql

MySQL索引的优化之LIKE模糊查询功能实现

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');

七、使用外部搜索引擎

对于复杂的模糊查询需求,尤其是大数据量场景,可以使用外部搜索引擎(如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%';

九、缓存结果

如果模糊查询的结果不经常变化,可以将查询结果缓存起来,减少数据库的查询压力

示例:

总结

通过以上方法,可以显著优化mysql中like模糊查询的性能。根据具体的业务需求和数据特点,选择合适的优化策略:

注:了解mysql-match ... against工具参考mysql-match ... against工具

到此这篇关于mysql--索引的优化--like模糊查询的文章就介绍到这了,更多相关mysql like模糊查询内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

您想发表意见!!点此发布评论

推荐阅读

MySql match against工具详细用法

04-21

Nginx配置文件的使用以及实现负载均衡的4种常用方式

04-22

nginx实现负载均衡与实例解读

04-22

利用Nginx实现资源代理和接口代理的实现方法

04-22

MySQL存储引擎InnoDB架构原理和执行流程

04-20

mysql中的group by高级用法

04-23

猜你喜欢

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论