like 和 contains 和match() against()

Like直接在数据据中查找可以查到所有所需记录但是会扫描整个表会影响性能,

CONTAINS是基于全文索引 进行查询,查询结果受系统全文索引分词的方法影响查询结果会不全。
Select * FROM A Where CONTAINS(B,'"IT"Or"理论"')5257条记录
Select * FROM A Where B Like'%IT%' or B LIKE '%理论%' 5468条记录
结论:需要精确查询用Like如产品搜索,内容搜索可以用CONTAINS提高效率。

 

全文索引——CONTAINS 语法
我们通常在 WHERE 子句中使用 CONTAINS ,就象这样:SELECT * FROM table_name WHERE CONTAINS(fullText_column,'search contents')。 php学习之家

我们通过例子来学习,假设有表 students,其中的 address 是全文本检索的列。
1. 查询住址在北京的学生 

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, 'beijing' )  

remark: beijing是一个单词,要用单引号括起来 。

2. 查询住址在河北省的学生 

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, '"HEIBEI province"' )  

remark: HEBEI province是一个词组,在单引号里还要用双引号括起来 。

3. 查询住址在河北省或北京的学生 

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, '"HEIBEI province" OR beijing' )  

remark: 可以指定逻辑操作符(包括 AND ,AND NOT,OR )。

4. 查询有 '南京路' 字样的地址 

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, 'nanjing NEAR road' )  

remark: 上面的查询将返回包含 'nanjing road','nanjing east road','nanjing west road' 等字样的地址。
A NEAR B,就表示条件: A 靠近 B。

5. 查询以 '湖' 开头的地址 

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, '"hu*"' )  

remark: 上面的查询将返回包含 'hubei','hunan' 等字样的地址。记住是 *,不是 %。 www.444p.com

6. 类似加权的查询

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, 'ISABOUT (city weight (.8), county wright (.4))')  

remark: ISABOUT 是这种查询的关键字,weight 指定了一个介于 0~1之间的数,类似系数(我的理解)。表示不同条件有不同的侧重。

7. 单词的多态查询 

Java代码  

  1. SELECT student_id,student_name FROM students WHERE CONTAINS( address, 'FORMSOF (INFLECTIONAL,street)' )  

remark: 查询将返回包含 'street','streets'等字样的地址。
对于动词将返回它的不同的时态,如:dry,将返回 dry,dried,drying 等等。

以上例子都使用英文,不使用中文是因为有的查询方式中文不支持 。

 

MATCH (col1,col2,...) AGAINST (expr [IN BOOLEAN MODE | WITH QUERY EXPANSION])
MySQL支持全文索引和搜索功能。MySQL中的全文索引类型FULLTEXT的索引。 FULLTEXT 索引仅可用于 MyISAM 表;他们可以从CHAR、 VARCHAR或TEXT列中作为CREATE TABLE语句的一部分被创建,或是随后使用ALTER TABLE 或 CREATE INDEX被添加。对于较大的数据集,将你的资料输入一个没有FULLTEXT索引的表中,然后创建索引, 其速度比把资料输入现有FULLTEXT索引的速度更为快。

Java代码  

  1. mysql> CREATE TABLE articles (  
  2. -> id INT UNSIGNED AUTO_INCREMENT NOT NULL PRIMARY KEY,  
  3. -> title VARCHAR(200),  
  4. -> body TEXT,  
  5. -> FULLTEXT (title,body)  
  6. -> );  
  7. Query OK, 0 rows affected (0.00 sec)  
  8.   
  9. mysql> INSERT INTO articles (title,body) VALUES  
  10. -> ('MySQL Tutorial','DBMS stands for DataBase ...'),  
  11. -> ('How To Use MySQL Well','After you went through a ...'),  
  12. -> ('Optimizing MySQL','In this tutorial we will show ...'),  
  13. -> ('1001 MySQL Tricks','1. Never run mysqld as root. 2. ...'),  
  14. -> ('MySQL vs. YourSQL','In the following database comparison ...'),  
  15. -> ('MySQL Security','When configured properly, MySQL ...');  
  16. Query OK, 6 rows affected (0.00 sec)  
  17. Records: 6 Duplicates: 0 Warnings: 0  
  18.   
  19. mysql> SELECT * FROM articles WHERE MATCH (title,body) AGAINST ('database' IN BOOLEAN MODE);  
  20. +----+-------------------+------------------------------------------+  
  21. | id | title | body |  
  22. +----+-------------------+------------------------------------------+  
  23. | 5 | MySQL vs. YourSQL | In the following database comparison ... |  
  24. | 1 | MySQL Tutorial | DBMS stands for DataBase ... |  
  25. +----+-------------------+------------------------------------------+  
  26. 2 rows in set (0.00 sec)  

 MATCH()函数对于一个字符串执行资料库内的自然语言搜索。一个资料库就是1套1个或2个包含在FULLTEXT内的列。搜索字符串作为对 AGAINST()的参数而被给定。对于表中的每一行, MATCH() 返回一个相关值,即, 搜索字符串和 MATCH()表中指定列中该行文字之间的一个相似性度量。

在默认状态下, 搜索的执行方式为不区分大小写方式。然而,你可以通过对编入索引的列使用二进制排序方式执行区分大小写的全文搜索。 例如,可以向一个使用latin1字符集的列给定latin1_bin 的排序方式,对于全文搜索区分大小写。

如上述所举例子,当MATCH()被用在一个 WHERE 语句中时,相关值是非负浮点数。零相关的意思是没有相似性。相关性的计算是基于该行中单词的数目, 该行中独特子的数目,资料库中单词的总数,以及包含特殊词的文件(行)数目。

对于自然语言全文搜索,要求MATCH() 函数中命名的列和你的表中一些FULLTEXT索引中包含的列相同。对于前述问讯, 注意,MATCH()函数(题目及全文)中所命名的列和文章淼腇ULLTEXT索引中的列相同。若要分别搜索题目和全文,应该对每个列创建 FULLTEXT索引。

上面的例子基本上展示了怎样使用返回行的相关性顺序渐弱的MATCH()函数。而下面的例子则展示了怎样明确地检索相关值。返回行的顺序是不定的,原因是 SELECT 语句不包含 WHERE或ORDER BY 子句:

Java代码  

  1. mysql> SELECT id, MATCH (title,body) AGAINST ('Tutorial')  
  2. -> FROM articles;  
  3. +----+-----------------------------------------+  
  4. | id | MATCH (title,body) AGAINST ('Tutorial' IN BOOLEAN MODE) |  
  5. +----+-----------------------------------------+  
  6. | 1 | 0.65545833110809 |  
  7. | 2 | 0 |  
  8. | 3 | 0.66266459226608 |  
  9. | 4 | 0 |  
  10. | 5 | 0 |  
  11. | 6 | 0 |  
  12. +----+-----------------------------------------+  
  13. 6 rows in set (0.00 sec)  

下面的例子则更加复杂。询问返回相关值,同时对行按照相关性渐弱的顺序进行排序。为实现这个结果,你应该两次指定 MATCH(): 一次在 SELECT 列表中而另一次在 WHERE子句中。这不会引起额外的内务操作,原因是MySQL 优化程序注意到两个MATCH()调用是相同的,从而只会激活一次全文搜索代码。

Java代码  

  1. mysql> SELECT id, body, MATCH (title,body) AGAINST  
  2. -> ('Security implications of running MySQL as root') AS score  
  3. -> FROM articles WHERE MATCH (title,body) AGAINST  
  4. -> ('Security implications of running MySQL as root' IN BOOLEAN MODE);  
  5. +----+-------------------------------------+-----------------+  
  6. | id | body | score |  
  7. +----+-------------------------------------+-----------------+  
  8. | 4 | 1. Never run mysqld as root. 2. ... | 1.5219271183014 |  
  9. | 6 | When configured properly, MySQL ... | 1.3114095926285 |  
  10. +----+-------------------------------------+-----------------+  
  11. 2 rows in set (0.00 sec)  
  12. 表中有2行(0.00 秒)  

 

再为表添加 fulltext 上去
mysql> alter table articles add fulltext index(title,body);

MySQL FULLTEXT 执行将任何单字字符原形 (字母、数字和下划线部分)的序列视为一个单词。这个序列或许也包含单引号 ('),但在一行中不会超过一个。 这意味着 aaa'bbb 会被视为一个单词,而 aaa''bbb则被视为2个单词。位于单词之前或其后的单引号会被FULLTEXT分析程序去掉; 'aaa'bbb' 会变成 aaa'bbb。

FULLTEXT分析程序会通过寻找某些分隔符来确定单词的起始位置和结束位置,例如' ' (间隔符号)、 , (逗号)以及 . (句号 )。假如单词没有被分隔符分开,(例如在中文里 ), 则 FULLTEXT 分析程序不能确定一个词的起始位置和结束位置。为了能够在这样的语言中向FULLTEXT 索引添加单词或其它编入索引的术语,你必须对它们进行预处理,使其被一些诸如"之类的任意分隔符分隔开。

一些词在全文搜索中会被忽略:

任何过于短的词都会被忽略。 全文搜索所能找到的词的默认最小长度为 4个字符。
停止字中的词会被忽略。禁用词就是一个像“the” 或“some” 这样过于平常而被认为是不具语义的词。存在一个内置的停止字, 但它可以通过用户自定义列表被改写。

词库和询问中每一个正确的单词根据其在词库和询问中的重要性而被衡量。 通过这种方式,一个出现在许多文件中的单词具有较低的重要性(而且甚至很多单词的重要性为零),原因是在这个特别词库中其语义价值较低。反之,假如这个单 词比较少见,那么它会得到一个较高的重要性。然后单词的重要性被组合,从而用来计算该行的相关性。

这项技术最适合同大型词库一起使用 (事实上, 此时它经过仔细的调整 )。对于很小的表,单词分布并不能充分反映它们的语义价值, 而这个模式有时可能会产生奇特的结果。例如, 虽然单词 “MySQL” 出现在文章表中的每一行,但对这个词的搜索可能得不到任何结果:

Java代码  

  1. mysql> SELECT * FROM articles WHERE MATCH (title,body) AGAINST ('MySQL');  

 找不到搜索的词(0.00 秒)

这个搜索的结果为空,原因是单词 “MySQL” 出现在至少全文的50%的行中。 因此, 它被列入停止字。对于大型数据集,使用这个操作最合适不过了----一个自然语言问询不会从一个1GB 的表每隔一行返回一次。对于小型数据集,它的用处可能比较小。 

一个符合表中所有行的内容的一半的单词查找相关文档的可能性较小。事实上, 它更容易找到很多不相关的内容。我们都知道,当我们在因特网上试图使用搜索引擎寻找资料的时候,这种情况发生的频率颇高。可以推论,包含该单词的行因其所 在特别数据集 而被赋予较低的语义价值。 一个给定的词有可能在一个数据集中拥有超过其50%的域值,而在另一个数据集却不然。

当你第一次尝试使用全文搜索以了解其工作过程时,这个50% 的域值提供重要的蕴涵操作:若你创建了一个表,并且只将文章的1、2行插入其中, 而文中的每个单词在所有行中出现的机率至少为 50% 。那么结果是你什么也不会搜索到。一定要插入至少3行,并且多多益善。需要绕过该50% 限制的用户可使用布尔搜索代码

 

全文索引效率确实高,但是也产生了很多不符合条件的记录。like则效率低下分词有多种方法 
比如把 "中华人民共和国 "分词简单的就有如下两种: 
1、中华   人民   共和国 
2、中华   华人   人民   民共   共和   和国

时间: 2024-10-18 02:05:07

like 和 contains 和match() against()的相关文章

Description Resource Path Location Type Java compiler level does not match the version of the instal

Description Resource Path Location Type Java compiler level does not match the version of the instal 解决办法 在项目上右键Properties->Project Facets,在打开的Project Facets页面中的Java下拉列表中,选择相应版本. 有可能是java1.6 改成java6之类的

主备不一致:Table definition on master and slave does not match

昨天一同事在线上做变更,为了保证主库的稳定性,先在备库把binlog关闭,然后在进行DDL变更,在通过切换HA,把备库切换为主库,在老的主库上做DDL变更 看上去这样做法没有太大的问题,但是当备库变更一做完,HA切换到备库,开始老主库变更的时候,备库就出现复制出现错误: Last_Error: Table definition on master and slave does not match: Column 10 type mismatch – received type 3, dbname

match 方法

  使用正则表达式模式对字符串执行查找,并将包含查找的结果作为数组返回. stringObj.match(rgExp) 参数 stringObj 必选项.对其进行查找的 String 对象或字符串文字. rgExp 必选项.为包含正则表达式模式和可用标志的正则表达式对象.也可以是包含正则表达式模式和可用标志的变量名或字符串文字. 说明 如果 match 方法没有找到匹配,返回 null.如果找到匹配返回一个数组并且更新全局 RegExp 对象的属性以反映匹配结果. match 方法返回的数组有三

EXCEL下使用MATCH函数碰到的问题

excel|函数|问题 Sub USEMATCH()   Dim s_p As String, e_p As String   Dim num As Integer    num = 0    For Each M In Range("a:a")      If M.Value <> "" Then        num = num + 1      Else        Exit For      End If    Next M    erange

F#教程:Pattern Match

这回我们要挑战下match语句: let s = "red" match s with | "red" -> printfn "红" | "white" -> printfn "白" | _ -> printfn "其他颜色" 其中,分支的顺序很重要. _ 可以匹配一切.如果把它写在最开始,其他的条件就不会被匹配到了. 但是,如果写成如下这样: let n = 2 ma

C#正则表达式类Match和Group类的理解

正则表达式可以看做一种有特定功能的小型编程语言,在一段文本中定位子字符串.利用正则表达式可以快速地分析大量的文本以找到特定的字符模式:提取.编辑.替换或删除文本子字符串:或将提取的字符串添加到集合.正则表达式的基本语法可参见:深入浅出之正则表达式(一),深入浅出之正则表达式(二). C#命名空间System.Text.RegularExpressions提供了支持正则表达式操作的类.这些类主要包括Regex,MatchCollection,Match,GroupCollection,Group,

python正则表达式match和search用法实例

 这篇文章主要介绍了python正则表达式match和search用法,实例分析了正则表达式中match和search的功能.定义及相关使用技巧,具有一定参考借鉴价值,需要的朋友可以参考下     本文实例讲述了python正则表达式match和search用法.分享给大家供大家参考.具体分析如下: python提供了2中主要的正则表达式操作:re.match 和 re.search. match :只从字符串的开始与正则表达式匹配,匹配成功返回matchobject,否则返回none: sea

ZBrush中Match Maker匹配笔刷使用指南

  步骤一 在Tool[工具]菜单中选择Sphere3D物体,在视图中拖动鼠标左键创建球体,按下视图顶部工具架中的Edit[编辑]按钮,进入3D编辑模式. 步骤二 单击Tool:Make PolyMesh3D[工具:创建3D多变形网格]命令,将参数化球体转换为多边形网格,如下图所示. 步骤三 单击Geometry[几何体]面板中的Divide[细分]按钮,将模型细分成5级. 步骤四 按住Ctrl和Shift键的同时,单击左工具架最上端的雕刻笔刷图标,弹出浮动面板,选择ClipCurve笔刷,然后

wps表格如何使用index和match函数

wps表格使用index和match函数的方法: 1.我们编辑了一个公司人员出差的大概的费用表,我们来选择查询使用. 2.这个时候我们输入我们的函数:第一个是要我们输入想要查询的数组 3.选择里面的数值数据,这里使用match函数挑选出选中的数据. 4.后面在选择要使用match函数查找的区域.这里我们使用精确查找. 5.这时候我们移出光标到后面输入英文逗号,再输入match查找另一个数据.格式基本上相同. 6.最后我们完成的函数如下图所示,函数是:=INDEX(I23:J25,MATCH(H3

JavaScript中String.match()方法的使用详解

  这篇文章主要介绍了JavaScript中String.match()方法的使用详解,是JS入门学习中的基础知识,需要的朋友可以参考下 此方法用于当匹配针对正则表达式的字符串来检索匹配. 语法 ? 1 string.match( param ) 下面是参数的详细信息: param : 正则表达式对象 返回值: 如果正则表达式不包括g标志,返回的结果相同于regexp.exec(string) 如果正则表达式包含g标志,则该方法返回一个包含所有匹配的数组 例子: ? 1 2 3 4 5 6 7