浅析MySQL模式匹配查询(标准SQL模式匹配和正则模式匹配)

在业务开发中,经常有需要模糊匹配某个字段的需求。比如按某个名字匹配,但用户可能只记得部分字,没记住全名,如果能支持模糊匹配那用户体验就会好很多。

MySQL 提供了标准SQL模式匹配以及一种基于扩展正则表达式的模式匹配,类似于 Unix 实用程序(如 vi、grep和 sed )使用的那些。

SQL模式匹配

SQL模式匹配使您能够用”_”匹配任何单个字符,使用”%”匹配任意数量的字符(包括零个字符)。

在MySQL中,SQL模式默认不区分大小写。不要在使用SQL模式时使用=或<>,请改用LIKE或者NOT LIKE运算符。

要查找以b开头的名称:

mysql>SELECT*FROMpetWHEREnameLIKE'b%';
+--------+--------+---------+------+------------+------------+
|name|owner|species|sex|birth|death|
+--------+--------+---------+------+------------+------------+
|Buffy|Harold|dog|f|1989-05-13|NULL|
|Bowser|Diane|dog|m|1989-08-31|1995-07-29|
+--------+--------+---------+------+------------+------------+

查找以fy结尾的名称:

mysql>SELECT*FROMpetWHEREnameLIKE'%fy';
+--------+--------+---------+------+------------+-------+
|name|owner|species|sex|birth|death|
+--------+--------+---------+------+------------+-------+
|Fluffy|Harold|cat|f|1993-02-04|NULL|
|Buffy|Harold|dog|f|1989-05-13|NULL|
+--------+--------+---------+------+------------+-------+

要查找包含w的名称:

mysql>SELECT*FROMpetWHEREnameLIKE'%w%';
+----------+-------+---------+------+------------+------------+
|name|owner|species|sex|birth|death|
+----------+-------+---------+------+------------+------------+
|Claws|Gwen|cat|m|1994-03-17|NULL|
|Bowser|Diane|dog|m|1989-08-31|1995-07-29|
|Whistler|Gwen|bird|NULL|1997-12-09|NULL|
+----------+-------+---------+------+------------+------------+

要查找恰好包含五个字符的名称,请使用模式字符”_”:

mysql>SELECT*FROMpetWHEREnameLIKE'_____';
+-------+--------+---------+------+------------+-------+
|name|owner|species|sex|birth|death|
+-------+--------+---------+------+------------+-------+
|Claws|Gwen|cat|m|1994-03-17|NULL|
|Buffy|Harold|dog|f|1989-05-13|NULL|
+-------+--------+---------+------+------------+-------+

正则模式匹配

在通常情况下,上述功能已经可以满足大部分业务的开发需求。但偶尔也会遇到一些更复杂的查询匹配需求,可能需要正则才能满足。MySQL 提供的另一种模式匹配使用扩展的正则表达式。当您测试这种类型的模式是否匹配时,请使用REGEXP和NOT REGEXP运算符(或 RLIKE与NOT RLIKE,它们是同义词)。

个人比较偏爱RLIKE、NOT RLIKE,与LIKE、NOT LIKE类似,好记。

  • . 匹配任何单个字符。
  • 字符类 […] 匹配括号内的任何字符。例如, [abc]匹配a、 b或c。要命名一系列字符,请使用破折号,[a-z] 匹配任何字母,而[0-9] 匹配任何数字。
  • * 匹配它前面的事物的零个或多个实例。例如,x* 匹配任意数量的x字符、 [0-9]*匹配任意数量的数字以及.*匹配任意数量的任何内容。
  • 如果模式匹配正在测试的值中的任何位置,则正则表达式模式匹配成功。(这与LIKE模式匹配不同,LIKE模式匹配只有在模式匹配整个值时才会成功。)
  • 要锚定一个模式,使其必须匹配被测试值^的开头或$结尾,请在模式的开头或结尾使用。

为了演示扩展正则表达式是如何工作的,LIKE前面显示的查询在这里用正则重写。

要查找以b开头的名称,使用^匹配名称的开头:

mysql>SELECT*FROMpetWHEREnameREGEXP'^b';
+--------+--------+---------+------+------------+------------+
|name|owner|species|sex|birth|death|
+--------+--------+---------+------+------------+------------+
|Buffy|Harold|dog|f|1989-05-13|NULL|
|Bowser|Diane|dog|m|1989-08-31|1995-07-29|
+--------+--------+---------+------+------------+------------+

要强制REGEXP比较区分大小写,请使用BINARY关键字使其中一个字符串成为二进制字符串。此查询仅匹配b名称开头的小写字母:

SELECT*FROMpetWHEREnameREGEXPBINARY'^b';

要查找以fy结尾的名称,使用$匹配名称的结尾:

mysql>SELECT*FROMpetWHEREnameREGEXP'fy$';
+--------+--------+---------+------+------------+-------+
|name|owner|species|sex|birth|death|
+--------+--------+---------+------+------------+-------+
|Fluffy|Harold|cat|f|1993-02-04|NULL|
|Buffy|Harold|dog|f|1989-05-13|NULL|
+--------+--------+---------+------+------------+-------+

要查找包含w的名称,请使用以下查询:

mysql>SELECT*FROMpetWHEREnameREGEXP'w';
+----------+-------+---------+------+------------+------------+
|name|owner|species|sex|birth|death|
+----------+-------+---------+------+------------+------------+
|Claws|Gwen|cat|m|1994-03-17|NULL|
|Bowser|Diane|dog|m|1989-08-31|1995-07-29|
|Whistler|Gwen|bird|NULL|1997-12-09|NULL|
+----------+-------+---------+------+------------+------------+

因为正则表达式模式会匹配出现在值的任何位置,因此在前面的查询中,没有必要像SQL 模式那样在模式的任一侧放置通配符以使其匹配整个值。

要查找恰好包含五个字符的名称,使用^、$、. 来匹配名称的开头和结尾:

mysql>SELECT*FROMpetWHEREnameREGEXP'^.....$';
+-------+--------+---------+------+------------+-------+
|name|owner|species|sex|birth|death|
+-------+--------+---------+------+------------+-------+
|Claws|Gwen|cat|m|1994-03-17|NULL|
|Buffy|Harold|dog|f|1989-05-13|NULL|
+-------+--------+---------+------+------------+-------+

还可以使用{n} (“repeat-n-times”)运算符:

mysql>SELECT*FROMpetWHEREnameREGEXP'^.{5}$';
+-------+--------+---------+------+------------+-------+
|name|owner|species|sex|birth|death|
+-------+--------+---------+------+------------+-------+
|Claws|Gwen|cat|m|1994-03-17|NULL|
|Buffy|Harold|dog|f|1989-05-13|NULL|
+-------+--------+---------+------+------------+-------+

注意事项

两种模式匹配区别

Like匹配原则是要求模式串与整个目标字段匹配时,才返回该条记录。

正则匹配则是当目标字段包含模式串时即返回该条记录。

模式匹配可能无法走索引

InnoDB 在模糊查询数据时使用 “%xx” 会导致索引失效,正则也类似。

最好能结合其他索引字段一起查询,这样效率比较高。

如果数据比较多,建议用全文索引。

正则匹配注意特殊字符

正则匹配功能强大,但使用时一定需要注意特殊字符的干扰,尤其是在mysql中查中括号或者小括号,一定要转义:

mysql>SELECTREGEXP_LIKE('(', '(');
ERROR3692 (HY000): Mismatchedparenthesisinregularexpression.
mysql>SELECTREGEXP_LIKE('(', '\\(');
+-------------------------+
|REGEXP_LIKE('(', '\\(') |
+-------------------------+
|1|
+-------------------------+
mysql>SELECTREGEXP_LIKE(')', ')');
ERROR3692 (HY000): Mismatchedparenthesisinregularexpression.
mysql>SELECTREGEXP_LIKE(')', '\\)');
+-------------------------+
|REGEXP_LIKE(')', '\\)') |
+-------------------------+
|1|
+-------------------------+

mysql>SELECTREGEXP_LIKE('[', '[');
ERROR3696 (HY000): Theregularexpressioncontainsan
unclosedbracketexpression.
mysql>SELECTREGEXP_LIKE('[', '\\[');
+-------------------------+
|REGEXP_LIKE('[', '\\[') |
+-------------------------+
|1|
+-------------------------+
mysql>SELECTREGEXP_LIKE(']', ']');
+-----------------------+
|REGEXP_LIKE(']', ']') |
+-----------------------+
|1|
+-----------------------+

解决办法:

  1. 限制使用的字符范围,排除元字符。
  2. 查询之前先转义,例如golang中,使用func QuoteMeta(s string) string
QuoteMetareturnsastringthatescapesallregularexpressionmetacharactersinsidetheargumenttext; thereturnedstringisaregularexpressionmatchingtheliteraltext.

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/306467.html<

(0)
运维的头像运维
上一篇2025-05-26 16:21
下一篇 2025-05-26 16:23

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注