Mysql8 %中间%模糊搜索 强行 支持 使用 索引 代替全文搜索 全文检索数据库

用户表有个昵称字段需要模糊搜索 而且搜索量巨大 而且必须支持特殊字符串

个人不喜欢架构复杂化 多用elasticsearch、solr、sphinx 很麻烦

所以研究了个通用解决方案 mysql支持索引模糊搜索

1.首先修改mysql配置 改成全文索引最小为1字符串

sql 复制代码
[mysqld]  
innodb_ft_min_token_size=1
ft_min_word_len=1

2.新建STORED 虚拟字段 nickname_index 绑定原始nickname字段变化跟着一起变

3.nickname_index 创建全文索引

sql 复制代码
CREATE TABLE `tb_user` (
  `uid` int unsigned NOT NULL AUTO_INCREMENT COMMENT '用户ID',
  `nickname` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_bin NOT NULL DEFAULT '' COMMENT '别名',
  `nickname_index` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin GENERATED ALWAYS AS (regexp_replace(`nickname`,_utf8mb4'(.)',_utf8mb4'$1 ')) STORED COMMENT '全文索引',

  PRIMARY KEY (`uid`) USING BTREE,
  FULLTEXT KEY `nickname_index` (`nickname_index`)
) ENGINE=InnoDB AUTO_INCREMENT=1001371 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin COMMENT='用户表';

4.客户端写SQL注意事项 需要用同样的方案

sql 复制代码
desc
SELECT 
* 
from tb_user 
where 

MATCH(nickname_index) AGAINST( regexp_replace('关键字','(.)','$1 ') )  

and 

nickname like CONCAT('%','关键字','%') 

成功使用索引了

实现方案是

1.用虚拟键绑定原始键(原始值变了 索引也会跟着自动变)

2.利用全文索引 (把虚拟键字符串逐字插入空格 强制用1字符串分词)

sql 复制代码
 regexp_replace('关键字','(.)','$1 ')

3.使用时先用全文索引 搜索分好词的关键字 再like一下原始键 数据就准确了

sql 复制代码
where 

MATCH(nickname_index) AGAINST( regexp_replace('关键字','(.)','$1 ') )  

and 

nickname like CONCAT('%','关键字','%') 

PS: 低版本mysql 没有regexp_replace 需要用编程语言先处理好再传进来 也可以支持 麻烦点

作者:LingMax 原创mysql支持索引模糊搜索 喜欢点个赞

相关推荐
不会C++的雾19 分钟前
Linux操作系统(2)
linux·数据库·mysql
zqmattack1 小时前
SQL优化与索引策略实战指南
java·数据库·sql
lang201509282 小时前
Jackson 1.x到2.x的演进与Spring集成
数据库·sql·spring
我星期八休息2 小时前
MySQL数据可视化实战指南
数据库·人工智能·mysql·算法·信息可视化
highly20092 小时前
Gitflow
大数据·elasticsearch·搜索引擎
码农幻想梦3 小时前
实验四 mybatis动态sql及逆向工程
sql·性能优化·mybatis
五阿哥永琪3 小时前
MySQL面试题 事务的隔离级别
数据库·mysql
不染尘.3 小时前
Linux的rpm与yum
linux·mysql·jdk·centos·tomcat·ssh
妄汐霜4 小时前
小白学习笔记(MySQL基础中其他知识)
笔记·学习·mysql
jay神4 小时前
基于Java的水果网上订购平台
java·mysql·vue·springboot·计算机毕业设计