SQL中对字符串字段模糊查询(LIKE)的索引命中情况

SQL中对字符串字段模糊查询(LIKE)的索引命中情况

在SQL数据库中,字符串字段上的索引是否会被LIKE查询使用取决于多种因素,包括数据库类型、索引结构、查询模式以及数据库配置。

主要数据库中的行为

1. MySQL/MariaDB

  • 前缀匹配(pattern%):可以使用B-tree索引

  • 后缀匹配(%pattern) 包含匹配(%pattern%):不能使用标准索引

  • 特殊技巧

    sql 复制代码
    -- 使用反向索引处理后缀查询
    ALTER TABLE users ADD COLUMN name_reverse VARCHAR(255);
    CREATE INDEX idx_name_reverse ON users(name_reverse);
    SELECT * FROM users WHERE name_reverse LIKE REVERSE('%son');

2. PostgreSQL

  • 前缀匹配:可以使用B-tree索引

  • 复杂模式 :需使用pg_trgm扩展

    sql 复制代码
    CREATE EXTENSION pg_trgm;
    CREATE INDEX idx_name_trgm ON users USING gin(name gin_trgm_ops);

3. SQL Server

  • 前缀匹配:可以使用索引

  • 包含匹配 :可以尝试使用全文索引(FULLTEXT)

    sql 复制代码
    CREATE FULLTEXT INDEX ON users(name);

4. Oracle

  • 前缀匹配:可以使用索引
  • 复杂模式:可以使用Oracle Text功能

通用优化建议

  1. 尽量使用前缀匹配LIKE 'abc%'LIKE '%abc'效率高

  2. 考虑使用函数索引(如反向索引)

  3. 对于复杂搜索

    • 使用专门的全文搜索引擎(如Elasticsearch)
    • 使用数据库特定的全文搜索功能
  4. 检查执行计划

    sql 复制代码
    EXPLAIN SELECT * FROM table WHERE column LIKE 'pattern%';

何时索引会被使用

查询模式 是否可能使用索引
LIKE 'abc%' ✅ 是
LIKE '%abc' ❌ 否
LIKE '%abc%' ❌ 否
LIKE 'abc' (无通配符) ✅ 是(等同于=)

高级解决方案

  1. 使用计算列+索引

    sql 复制代码
    ALTER TABLE users ADD name_length AS LENGTH(name);
    CREATE INDEX idx_name_length ON users(name_length);
  2. 使用分区表:按字符串前缀分区

  3. 使用专门的搜索引擎:如Elasticsearch、Solr等

记住,索引的使用还取决于数据库优化器的决策,实际应以EXPLAIN分析结果为准。

相关推荐
键盘上的猫头鹰6 小时前
【从零学MySQL(三)】数据增删改(DML)及 SELECT 查询详解
数据库·mysql·数据分析
KaMeidebaby7 小时前
卡梅德生物技术快报|蛋白的过表达质粒构建与生信分析实验全流程复盘
前端·数据库·其他·百度·新浪微博
渣渣盟7 小时前
数据库之两段锁协议相关理论及应用
数据库·关系规范化·两段锁协议
LCG元7 小时前
Istio - 服务网格流量治理深度解析:灰度发布 / 故障注入配置实践
java·数据库·istio
SuniaWang7 小时前
《Agentx专栏》03-架构设计:AgentX的六层架构是如何生长出来的
java·数据库·redis·docker·ai·架构
键盘上的猫头鹰7 小时前
【从零学MySQL(二)】数据库基础操作、数据类型与约束(附Navicat演示)
数据库·mysql·数据分析·数据可视化
东风破1377 小时前
达梦DEM和DFM的介绍、搭建学习记录
数据库·学习·dm达梦数据库
小江的记录本7 小时前
【Kafka核心】Kafka 3.0+ KRaft模式(替代ZooKeeper)核心原理与优势
java·数据库·分布式·后端·zookeeper·kafka·rabbitmq
treacle田8 小时前
达梦数据库-数据库主备集群更改实例目录及相关目录步骤-记录总结
数据库·达梦数据库主备集群更改实例目录
hef2888 小时前
SQL和Python怎么选?数据分析工具实战指南
python·sql·数据分析