简单查询lite search (字符串查询)是一种有效的命令行ad hoc 查询,但是想要善用搜索,必须使用请求体查询request body search API.之所以这么称呼,是因为大多数的参数以JSON格式所容纳,而不是查询字符串.
请求体查询不但可以处理查询,而且还可以高亮返回结果中的片段.
GET _search
{}
同字符串查询一样,你可以查询一个,或多个索引及类型
GET /index_2014*/type1,type2/_search
{}
也可以使用from, size参数进行分页pagination:
GET /website/_search
{
"from":1,
"size":3
}
注意,from和size数值均可以与实际不符,返回的不过是个空的数组,并不会出错.
那么,这中请求体查询,使用的是携带内容的GET请求方式?
任何一种语言(特别是js)的HTTP库都不允许GET请求中携带交互数据,用户会很惊讶GET请求会允许携带交互数据.
但是真实情况是,一份关于HTTP协议的标准文档RFC中并未定义一个GET请求携带请求体会发生什么!所以,
ES的作者们倾向于使用GET提交查询请求,因为它们觉得这个词相比于POST能更好的描述这种行为.然而,因为携带请求体的GET请求并不被广泛支持,所以search API同样支持POST请求.
POST /website/_search
{
"from":1,
"size":3
}
相比于神秘的字符串查询方式,请求体查询允许我们通过使用query DSL(Domian Specific Language)来写入参数.
query DSL是一种灵活的,表现力强的查询语言,ES通过一个简单的JSON接口使用DSL来表现lucene绝大多数的能力.
应当在你的产品中使用这种方式进行查询,它是你的查询更加灵活,精准,易于阅读,且易于debug.
为了使用query DSL,传递一个查询给query参数:
GET /_search
{
"query": YOUR_QUERY_HERE
}
例如,空查询,其实就相当于使用了一个match_all查询子句
POST /website/_search
{
"query": {
"match_all": {}
}
}
match_all是一个查询子句,正如其名字一样,查询所有文档.
查询子句的结构
一个查询子句的典型结构:
{
QUERY_NAME: {
ARGUMENT: VALUE,
ARGUMENT: VALUE,...
}
}
如果它是与特定字段有关的:
{
QUERY_NAME: {
FIELD_NAME: {
ARGUMENT: VALUE,
ARGUMENT: VALUE,...
}
}
}
例如,你可以使用match查询子句,查询在字段tweet中有elasticsearch的:
查询自己的格式:
{
"match": {
"tweet": "elasticsearch"
}
}
请求:
GET /_search
{
"query": {
"match": {
"tweet": "elasticsearch"
}
}
}
称之为查询子句query clause,代表其都是放到query语句下的.
合并多子句
查询子句像一个简单的积木块一样,可以和其他的子句组合,构成复杂的查询.
子句可以分为:
叶子子句leaf clause,被用作字符串与字段的比较.
复合子句compound clause,备用做合并其他的子句.例如一个bool子句,允许合并其他的子句:must 匹配, must_not,should.它还允许包含non-scoring, filters作为结构化搜索:
{
"bool": {
"must": { "match": { "tweet": "elasticsearch" }},
"must_not": { "match": { "name": "mary" }},
"should": { "match": { "tweet": "full text" }},
"filter": { "range": { "age" : { "gt" : 30 }} }
}
}
非常重要的指出,一个复合查询子句可以包含其他任何查询子句,或者别的复合子句.这意味着复合子句可以被相互嵌套,允许复杂的逻辑表达.
例如,下边的例子,查询邮件,满足:包含business opportunity,同时被标星的邮件;或者同时在folder,indbox,但是没有被标记为spam的邮件.
{
"bool": {
"must": { "match": { "email": "business opportunity" }},
"should": [
{ "match": { "starred": true }},
{ "bool": {
"must": { "match": { "folder": "inbox" }},
"must_not": { "match": { "spam": true }}
}}
],
"minimum_should_match": 1
}
}
不要担心这些例子的细节,我们后续会解释.重点是明白复合语句可以组合多个子句,包括叶子子句或这复合子句到一个简单的查询中.
ES使用DSL将查询子句放到一个简单的集里,这种简单集合可以被用作两种环境:过滤上下文Filtering context和查询上下文query context.
当被用到过滤环境中,查询query被称作non-scoring or filtering query,这样的查询会这样问问题,'这个文档是否匹配?'答案是二选一,是或否.
例如;
created 的日期范围是否介于2013-2014?
status字段是否包含词published?
las_lon字段的地理位置是否与目标相距不超过10km?
当被使用在查询环境中,查询成为scoring query,它这样问"这个文档的匹配程度如何?"
查询典型的使用:
查找与full text search 最佳匹配的文档
包含单词run,也可能是running,runs,jog, sprint
同时包含quick , brown, fox,它们离得越近,文档的匹配相关性越高.
标记着lucene, search, java,标识词越多,文档的相关性越高.
一个scoring query,计算文档与查询的相关性,并赋值给字段_score,用作依据相关性排序的标准.这种概念同样适用于全文搜索.
注意:
历史上,在ES中,查询和过滤是分开做的,在ES2.0开始,过滤被技术性的消除,同时,查询开始支持non-scoring式的查询.
然而为了区分和简便,我们仍用"过滤"一词来描述non-socring的查询.你可以把filter , filter query , non-scoring query当作一样的.
同样的,如果查询一词被单独的使用,我们就认为是scoring的查询.
性能差异
过滤查询是一个简单的包含与不包含的检查,这是它们计算非常快速.
有各种优化,对于至少有一条过滤查询是很少有文档匹配,同时被频繁的用作non-scoring的查询,可以被放到内存中,更快速获取.
相比之下,scoring查询不但需要查找匹配的文档,并且还要计算相关性,这使得其繁重于non-scoring查询,同时查询的结果是不能够被缓存的.
幸亏有倒排索引,使得一个简单的scoring查询,仅匹配一些文档,性能可以与过滤相比,甚至优于过滤,在跨越数以百万计的文件中.
但是一般情况下,过滤是优于查询的.
过滤的目的是减少文档的数量,这些文档必须被scoring query检查.
什么时候使用?
一般原则,在全文查询,或者需要相关性评分时,使用查询scoring query,其他时候都是使用过滤non-scoring query.
ES有很多查询语句,只有少部分经常被使用,我们会在后续的深入查询一章详细学习,现在快速介绍一些重要的语句.
match_all
match_all查询简单的匹配所有文档
{ "match_all": {}}
这个查询经常和过滤器一起使用.
match
match查询是一个标准的查询,无论是查询一个全文文本还是精确值.
如果使用match对全文文本字段进行查询,执行查询之前,先使用针对该字段正确的分析器对查询字符串进行分析.
{ "match": { "tweet": "About Search" }}
如果使用该语句在一个字段上匹配精确值,数值,日期,布尔,以及not_analyzed字符串,
{ "match": { "age": 26 }}
{ "match": { "date": "2014-09-01" }}
{ "match": { "public": true }}
{ "match": { "tag": "full_text" }}
对于确切值搜索,你可能想使用过滤语句,而不是查询,我们很快看到过滤的例子.
相比于字符串查询,match语句查询的语法更加安全.
multi_match
multi_match查询允许在多个字段上进行match一样的查询
{
"multi_match": {
"query": "full text search",
"fields": [ "title", "body" ]
}
}
range
gt : greater than
gte : greater than or equal to
lt : less than
lte : less than or equal to
{
"range": {
"age": {
"gte": 20,
"lt": 30
}
}
}
term
{ "term": { "age": 26 }}
{ "term": { "date": "2014-09-01" }}
{ "term": { "public": true }}
{ "term": { "tag": "full_text" }}
term查询不对输入的文本进行分析,所以它支持确切值查询
terms
terms查询同term查询,但是它允许指定多个匹配值,如果字段包含其中的任何一个,都会返回文档
{ "terms": { "tag": [ "search", "full_text", "nosql" ] }}
exist, missing
exist, missing查询被用作查询指定的字段存在的文档(exist)或者不存在的文档(missing),exist返回存在该字段的文档,missing返回不存在该字段的文档
{
"exists": {
"field": "title"
}
}
现实应用中的查询从来都不是简单的,使用多个输入值查询多个字段,依据一系列标准的过滤器.构造一个复杂查询,你需要一种组合多个查询子句在一个搜索请求中的方式.
为了达到这个要求,可以使用bool查询,这个查询接受如下参数:
must: 必须是匹配的文档被包含进来
must_not: 一定是不匹配的文档被包含进来
should: 如果匹配,增加_score,否则没有影响,为每个文档相关性评分.
filter: 必须匹配,是non_scoring的过滤模式,只是简单的包含或不包含.
因为这是我们看到的第一个包含其他查询的查询语句,我们需要谈论相关性评分是怎么计算的.
每个子句分别计算文档的相关性评分,一旦这些结果被计算出来,bool语句将这些分数合并到一起,并且返回一个单个分数值,代表bool操作的总分数.
接下来的查询,寻找文档:title字段匹配查询字符串"how to make millions",并且不被标识为spam.如果文档是starred,或者从2014开始的,它们的排名会比其他的文档高.
{
"bool": {
"must": { "match": { "title": "how to make millions" }},
"must_not": { "match": { "tag": "spam" }},
"should": [
{ "match": { "tag": "starred" }},
{ "range": { "date": { "gte": "2014-01-01" }}}
]
}
}
加上过滤查询:
如果我们不想文档的日期对评分产生影响,我们可以使用filter子句:
{
"bool": {
"must": { "match": { "title": "how to make millions" }},
"must_not": { "match": { "tag": "spam" }},
"should": [
{ "match": { "tag": "starred" }}
],
"filter": {
"range": { "date": { "gte": "2014-01-01" }}
}
}
}
通过将range查询放入filter子句,我们转化它为non-scoring查询,它不再对文档的相关性评分产生影响,并且因为是non-scoring查询,可以使用过滤器的优化来提升性能.
任何一个查询都可以使用这种方式,简单的将查询放到bool语句的filter子句中,会自动转化为non-scoring过滤.
如果需要一个基于多标准的过滤,bool查询本身可以作为non-scoring查询
{
"bool": {
"must": { "match": { "title": "how to make millions" }},
"must_not": { "match": { "tag": "spam" }},
"should": [
{ "match": { "tag": "starred" }}
],
"filter": {
"bool": {
"must": [
{ "range": { "date": { "gte": "2014-01-01" }}},
{ "range": { "price": { "lte": 29.99 }}}
],
"must_not": [
{ "term": { "category": "ebooks" }}
]
}
}
}
}
constant_score查询
尽管不如bool查询经常使用,constant_score查询也依然是有用的,该查询为匹配的文档应用静态的,常数的分数.它主要是在执行过滤查询时使用.
只有过滤子句时,你可以使用该语句代替bool语句.性能是相同的,但是有利于查询的简单性和清晰度
{
"constant_score": {
"filter": {
"term": { "category": "ebooks" }
}
}
}
查询可以是非常复杂,尤其是组合了不同的分析器和字段映射的时候,validate-query API可以检查一个请求是否有效.
在请求URL后加/_validate/query
GET /gb/tweet/_validate/query
{
"query": {
"tweet" : {
"match" : "really powerful"
}
}
}
validate请求的响应告诉我们请求是无效的:
{
"valid" : false,
"_shards" : {
"total" : 1,
"successful" : 1,
"failed" : 0
}
}
如果要知道问题处在了哪,可以在后边加上参数explain
GET /gb/tweet/_validate/query?explain
{
"query": {
"tweet" : {
"match" : "really powerful"
}
}
}
显然,我们混淆了查询语句的类别和字段的名字
{
"valid" : false,
"_shards" : { ... },
"explanations" : [ {
"index" : "gb",
"valid" : false,
"error" : "org.elasticsearch.index.query.QueryParsingException:
[gb] No query registered for [tweet]"
} ]
}
我们也可以利于expalin参数理解ES是如何解释查询的:
GET /us,gb/_validate/query?explain
{
"query": {
"match": {
"tweet": "really powerful"
}
}
}
为我们查询的每个索引返回一个explanation,因为每个索引有不同的映射和分析器:
{
"valid": true,
"_shards": {
"total": 2,
"successful": 2,
"failed": 0
},
"explanations": [
{
"index": "gb",
"valid": true,
"explanation": "tweet:realli tweet:power"
},
{
"index": "us",
"valid": true,
"explanation": "tweet:really tweet:powerful"
}
]
}
从explanation中,我们可以看出针对tweet字段,match语句是如何将查询字符串really powerful 重写为两个单个词term的.
两个索引的重写词不一样,原因是因为索引gb中的tweet字段使用的是english分析器.