面试--mysql的模糊查询优化、like、locate、position、instr、find_in_set

困扰很久的一个事情,一次面试的时候,在mysql的一个表中有个字段存的值是 “#床垫#白色#2*3” 类似的字段。
面试官让我匹配下 #白色 的所有数据,我当时第一反应就是通过 like 来处理,然后他说还有没有别的方法。
我想了下就说 find_in_set但是它的配置值必须要是用逗号隔开,所以应该不满足。

后面这个问题确实困扰了我,我就各种查找有效的方法,今天就总结下,MySQL的模糊查询,以后的多扩展下自己的思路,多尝试其他的方式,避免尴尬。

1) locate用法

locate(‘substr',str,pos)
SELECT LOCATE('xbar',`foobar`); 
返回0
SELECT LOCATE('bar',`foobarbar`); 
返回4
SELECT LOCATE('bar',`foobarbar`,5);
返回7

返回 substr 在 str 中第一次出现的位置,如果 substr 在 str 中不存在,返回值为 0 ;
如果pos存在,返回 substr 在 str 第pos个位置后第一次出现的位置;
如果 substr 在 str 中不存在,返回值为0。

SELECT `column` FROM `table` WHERE LOCATE('keyword', `field`) > 0
  • keyword是要搜索的内容,field为被匹配的字段,查询出所有存在keyword的数据

2) 其他的方式

  • POSITION('substr' IN field)方法
  • INSTR(str,'substr')方法

3)Like查询优化方案

like模糊查询形如'%AAA%'和'%AAA'将不会使用索引,但是业务上不可避免可能又需 要使用到这种形式查询方式:

  • 优化方案一:使用覆盖索引,即查询出的列只是用索引就可以获取,而无须查询表记录,这样也走了索引;

  • 优化方案二:使用locate函数或者position函数代替like查询:
    table.field like '%AAA%'可以改为locate('AAA', table.field) > 0POSITION('AAA' IN table.field)>0

4)使用正则表达式查询

使用 REGEXP 关键字指定正则表达式的字符匹配模式

正则表达式常用的字符匹配列表

选项 说明 例子 匹配值示例
^ 匹配文本的开始字符 ^b book,big,banana
$ 匹配文本结束字符 st$ test,resist
. 匹配任何单个字符 b.t bit,bat,but
* 匹配零个或多个在它前面的字符 f*n:匹配字符n前面的0个或多个f字符的字符串 fn,fan,faan
  • 匹配前面的字符1次或多次 ba+:匹配以 b 开头后面紧跟1个或多个a的字符串 ba,bay,bare
    <字符串> 匹配包含指定的字符串的文本 fa:匹配包含“fa”的字符串 fan,afa.faad
    [字符集合] 匹配字符集合中的任何一个字符 '[xz]':匹配 x 或者 z dizzy,zebra
    [^] 匹配不在括号中的任何字符
    '[^abc]':匹配任何不包含a、b、c的字符串

desk,fox
字符串{n,} 匹配前面的字符串至少n次 b{2}:匹配有2个或更多的b字符的字符串 bbb,bbbb
字符串{n,m} 匹配前面的字符串至少n次,至多m次。如果n为0,次参数为可选参数 b{2,4}:匹配至少有2个,最多有4个b字符的字符串
bb,bbb,bbbb

  • 查询以特定字符或字符串开头的记录
    字符^可以匹配以特定字符或者字符串开头的文本。
    【例】在 fruits 表中,查询 f_name 字段以字母 b 开头的记录。SQL 语句如下:

mysql> SELECT * FROM fruits WHERE f_name REGEXP '^b';

+------+------+------------+---------+
| f_id | s_id | f_name | f_price |
+------+------+------------+---------+
| b1 | 101 | blackberry | 10.20 |
| b2 | 104 | berry | 7.60 |
| t1 | 102 | blanana | 10.30 |
+------+------+------------+---------+

  • 查询以特定字符或字符串结尾的记录
    字符 $ 可以匹配以特定字符或者字符串结尾的文本。

    【例】在 fruits 表中,查询 f_name 字段以字母 y 结尾的记录。SQL 语句如下:

mysql> SELECT * FROM fruits WHERE f_name REGEXP 'y$';
+------+------+------------+---------+
| f_id | s_id | f_name | f_price |
+------+------+------------+---------+
| b1 | 101 | blackberry | 10.20 |
| b2 | 104 | berry | 7.60 |
| c0 | 101 | cherry | 3.20 |
| m2 | 105 | xbabay | 2.60 |
+------+------+------------+---------+

  • ** 代替字符串中的任意一个字符**
    字符‘.’可以匹配任意一个字符。

    【例】在 fruits 表中,查询 f_name 字段以包含字母 a 与 g 且两个字母之间只有一个字母的记录。SQL 语句如下

mysql> SELECT * FROM fruits WHERE f_name REGEXP 'a.g';
+------+------+--------+---------+
| f_id | s_id | f_name | f_price |
+------+------+--------+---------+
| bs1 | 102 | orange | 11.20 |
| m1 | 106 | mango | 15.60 |
+------+------+--------+---------+

  • 匹配多个字符
    星号(*) 可以任意次匹配前面的字符,包括 0 次。加号(+)至少匹配前面的字符一次。
    【例】在 fruits 表中,查询 f_name 字段以包含字母 b 开头,且 b 后面出现字母 a 的记录。SQL 语句如下:
    mysql> SELECT * FROM fruits WHERE f_name REGEXP '^ba*';
    +------+------+------------+---------+
    | f_id | s_id | f_name | f_price |
    +------+------+------------+---------+
    | b1 | 101 | blackberry | 10.20 |
    | b2 | 104 | berry | 7.60 |
    | t1 | 102 | banana | 10.30 |
    +------+------+------------+---------+

mysql> SELECT * FROM fruits WHERE f_name REGEXP '^ba+';
+------+------+--------+---------+
| f_id | s_id | f_name | f_price |
+------+------+--------+---------+
| t1 | 102 | banana | 10.30 |
+------+------+--------+---------+

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • 第5章 引用类型(返回首页) 本章内容 使用对象 创建并操作数组 理解基本的JavaScript类型 使用基本类型...
    大学一百阅读 3,231评论 0 4
  • 正则模式表: 代码示例:正则表达式可以与REGEXP 操作符一起使用 1、匹配字符串的开头(^):匹配name字段...
    阳光的微笑_f3cc阅读 3,301评论 0 1
  • 概要 64学时 3.5学分 章节安排 电子商务网站概况 HTML5+CSS3 JavaScript Node 电子...
    阿啊阿吖丁阅读 9,172评论 0 3
  • pyspark.sql模块 模块上下文 Spark SQL和DataFrames的重要类: pyspark.sql...
    mpro阅读 9,451评论 0 13
  • 这一篇最主要是记录下命令,方便以后查找 使用Mysql 创建数据库 create database mysql_t...
    Treehl阅读 576评论 0 0