ElasticSearch 7.4学习记录(DSL语法)

上文和大家一起初次了解了很多ES相关的基础知识,本文的内容将会是实际企业中所需要的吗,也是我们需要熟练应用的内容。
面对ES,我们最多使用的就是查询,当我负责这个业务时,现不需要我去考虑如何创建索引,添加文档等,只需要根据复杂业务实现查询即可,本文的重点也会在如何使用ES进行查询,并给出很多实际案例进行补充解释和演示

本次案例演示说明所需要的数据-----------student数据信息:

属性数值
id001
namezjh
address中国陕西省延安市
brief喜欢足球的男生
age28
属性数值
id002
namewxx
address中国陕西省渭南市
brief喜欢zjh的女生
age18

1 DSL查询语法

1.1 什么是DSL

就是在这个环境下查询语句,有自己的语法格式要求,需要我们熟练掌握。
或者你可以想一下,在mysql环境下的sql语法,这就是在ES环境下的DSL语法,都是为了实现查询功能。
在这里插入图片描述

一个简化的查询模版

GET /索引库名/_search
{
	"query":{
		"查询类型":{
			"查询条件":"条件值"
		}
	}
}

查询类型可以实现不同的查询效果,下面的案例很好的解释给了大家。

1.2 查询所有

match_all:查询全部的数据

GET /student/_search
{
	"query":{
		"match_all":{}
	}
}

1.3 全文检索

match查询:利用分词器进行查询,比如:传入“男生”,会将其分词取倒排索引库查询,找到含有“男”与“生”数据,即找到zjh和wxx的全部数据

这是一个模板
GET /student/_search
{
	"query":{
		"match":{
			"field":"text"
		}
	}
}

实际案例:拆分“男生”去brief字段中查找有无”男“与”生“,最终找到zjh和wxx
GET /student/_search
{
	"query":{
		"match":{
			"biref":"男生"
		}
	}
}

升级案例:现在不想指定一个字段去查找,想查询所有的字段中的数据是否有匹配的
GET /student/_search
{
	"query":{
		"match":{
			"all":"跑步工资"
		}
	}
}
这个all就表示会在所有字段id,name,address,brief的值中去匹配”跑步工资“

multi_match查询:允许同时查询多个字段

这是一个模板
GET /student/_search
{
	"query":{
		"multi_match":{
			"query":"text",
			"fields":["field1","field2"]
		}
	}
}
实际案例:找到address含有“延安”且brief含有”喜欢“的数据,最终找到zjh和wxx
GET /student/_search
{
	"query":{
		"multi_match":{
			"query":"延安喜欢",
			"fields":["address","brief"]
		}
	}
}

需要注意的是multi_match设置的字段越多,效率越慢,推荐优先使用match

1.4 精确查询

当查询条件是不可分的:keyword,数值,日期,boolean等,不会对其分词,即为精确查询

  • term:根据词条精确值查询
  • range:根据值范围查询

案例:
当查询内容传入:上海,就希望匹配到含有“上海”的数据,而不是含有“上”与“海”的这种数据。此乃term

当查询内容传入:100-300元,就希望匹配到在这个范围内的数据。此乃range

在这里插入图片描述

这是一个term模板
GET /student/_search
{
	"query":{
		"term":{
			"field":{
				“value”:“value”
			}
		}
	}
}
实际案例:找到address含有“男生”的数据,找到空数据,因为必须精准为男生,才能找到zjh
GET /student/_search
{
	"query":{
		"term":{
			"field":{
				“brief”:“男生延安”
			}
		}
	}
}
这是一个range模板
GET /student/_search
{
	"query":{
		"range":{
			"field":{
				“gte”:“value”,
				“lte”:“value”
			}
		}
	}
}
实际案例:找到年龄在10-20之间的人,最终找到wxx
GET /student/_search
{
	"query":{
		"range":{
			"age":{
				“gte”:10,
				“lte”:20
			}
		}
	}
}

注意:gte是大于等于;gt是大于

1.5 复合查询

Boolean Query
布尔查询是一个或者多个查询字句的组合。组合方式有

  • must : 必须匹配每个子查询(与)
  • should:选择性匹配子查询(或)
  • must_not:必须不匹配(非)不参与算分
  • filter:必须匹配 不参与算分
GET /student/_search
{
	"query":{
		“bool”:{
			“must”:[  
			       {"term":{“address”:“中国”} }
			       {"term":{“name”:“zjh”} }
			 ],
			 "should":[ 
			       {"term":{“brief”:“喜欢”} },
			       {"term":{“brief”:“足球”} }
			 ],
			 “must_not”:[
			 		{"range":{“age”:{"gte:20"} } },
			 ],
			 “filter”:[
			 		{"term":{“brief”:“生”}}
			 ]
		}
	}
}

这串代码表示:
must:(address必须含有中国) 且 (name必须含有zjh)
should:(brief含有喜欢) 或者 (brief含有足球)二选一
mustnot:(年龄大于等于20)相反-----(年龄小于20)

这个案例仅仅告诉你如何理解must等,下面介绍一个贴切的案例

查询名字包含“麦当劳”,地点在北京,人均消费不高于100元,周一到周天营业的快餐店

1.6 搜索结果处理

1.6.1 结果排序+分页查询

这是一个range模板
GET /student/_search
{
	"query":{
		“查询类型”:{}
		},
		"sort":[{"fileld":”value“}]
}
实际案例:全部查询,将结果按照年龄倒序,若相等按照id升序排
GET /student/_search
{
	"query":{
		"match_all":{}
		},
	"sort":[{”age“:”desc“},{”id“:”asc“}],
	“from”:0,
	“size”:10
		}
	}
}

es查询结果默认只显示10条数据。from:0--------size:10,表示从0开始,显示10条数据

实际上,如我们会将一批数据分成10批,存在10台服务器的ES上。如果我们需要查询价格由低到高排序前100条,底层执行的逻辑是:每一台ES由低到高排序,查出前100,然后聚合10台机器的数据(10*100),从聚合结果1000中再找到前100的数据。
不过呢,当我们执行数据读取时候,一定是会扫描所有的ES节点,不需要担心查询的时候会不会只查询当前服务器上es的数据,是不会的。

1.6.2 结果高亮

浏览器搜索java,可以看到查询结果中含有java部分高亮显示,这是如何实现的?
在这里插入图片描述
是ES帮忙做的,我们只需要告诉ES需要高亮显示的字段和内容即可

GET /student/_search
{
	"query":{
		“match”:{
			“all”:“中国”
		},
		“highlight”:{
			“fields”:{  //高亮字段
				“address”:{
					"requeire.field_match":“false”,//是否需要与查询字段匹配
					“pre_tags”:"<em>", //用来标记高亮字段的前置标签,
					“post_tags”:“</em>”  //用来标记高亮字段的后置标签
				}
			}
		}
}

2 DSL语法对应Java代码的实现

2.1 match_all

在这里插入图片描述

 RestHighLevelClient client 注意前面我已经获取了client
 关于代码实现,可以理解为四个部分,不同业务需要变动代码的部分只有2,4@Test
    void testMatchAll() throws IOException {
        //1.准备request
        SearchRequest request = new SearchRequest("student");

        //2.准备DSL
        request.source().query(QueryBuilders.matchAllQuery());

        //3.发送请求
        SearchResponse response = client.search(request, RequestOptions.DEFAULT);

        //4.解析响应中的hits
        SearchHits searchHits = response.getHits();

        long value = searchHits.getTotalHits().value; //获取value值(参考ES)
        System.out.println(value);

        SearchHit[] hits = searchHits.getHits(); //获取文档数据

        for (SearchHit hit : hits) {
            String json = hit.getSourceAsString();
            System.out.println(json);
        }

    }

为什么解析response呢,因为其全部内容如下图,我们需要选出需要的即可
在这里插入图片描述

2.1’ 核心代码梳理

这里梳理一下上述java代码的对应关系,帮助理解
source()方法中存在操作类型,有查询query,有结果排序sort,有分页form,size等
在这里插入图片描述

在这里插入图片描述

QueryBuilders()里面有查询类型,包括前面科普的精确查询term,全部查询match_all,复合查询bool等
在这里插入图片描述
在这里插入图片描述

2.2 match与multi_match

在这里插入图片描述

2.3 term和range

在这里插入图片描述

2.4 boolean

在这里插入图片描述

2.5 排序和分页

在这里插入图片描述

2.6 filter过滤查询

Query查询器的底层原理:首先需要深入理解query查询
在这里插入图片描述

这里我们的目的是,查询出来符合条件 taskTypeName= 文件服务器扫描 的数据。实际上对于query查询器来说,它会计算相关性得分,看存储的数据是否与taskTypeName= 文件服务器扫描相似
在这里插入图片描述
Filter查询的使用
constant_score:固定相似度得分,默认是1.0
boost:修改默认的相似度得分
在这里插入图片描述
在这里插入图片描述
可以看到,这里的数据相似度得分是我们指定的1.2。可以理解为我们使用filter过滤时,未对数据进行相似度计算,所以使用filter比query的性能高。

3 聚合索引

理解什么是聚合索引:类似于group by的概念,下图前两个常用。先看看聚合怎么用,文末会有个案例更清楚的解释聚合的概念
在这里插入图片描述

3.1 初体验聚合效果(buckets聚合案例)

比如我现在需要统计酒店的品牌数量有几种,则可以根据品牌数量进行聚合,就会得到名为“七天酒店” 34家 ;“如家” 30家 ;“速8” 20家的返回结果

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

3.2 优化聚合效果

对价格在200元以内的酒店进行聚合,这个案例表示query和aggs是平级。

在这里插入图片描述

3.3 Metrics聚合案例

在这里插入图片描述
也就是在Buckets聚合上再加了一层,很好记住,(sorce字段是酒店评分)注意观察下图,发现多了count、min、max、avg,sum。什么意思呢,就是说我们对酒店评分进行聚合,得到了最大值,最小值,平均值等等。
在这里插入图片描述

3.4 DSL语法对应Java代码的实现

在这里插入图片描述
在这里插入图片描述

3.5 多条件的聚合

就是同时聚合多个字段:城市、星级、品牌
在这里插入图片描述
如何理解这个业务
前面提到的精确查询,根据上海就可以查出city字段中含有”上海“的所有文档,但是我们仅仅停留在粗暴的找出信息的阶段,若我需要对上海这一类文档求平均价格,计数等,term的功能就显得捉襟见肘了。所以这里需要聚合处理。理解了业务需求在看看如何实现

1.准备request
2.准备DSL
	2.1 多个聚合字段
3.发出请求
4.解析结果
	4.1 分别处理聚合名称的数据

2.1内容
在这里插入图片描述
4.1内容,有几个集合字段就实现几次
在这里插入图片描述
实际上,你只需要清楚一个字段的聚合实现流程,那么当你写多个字段的聚合时,仅仅需要重复2和4的代码,那这些代码你可以封装成函数更优雅。

3.6 理解聚合

现在我们需要搜索城市是上海的数据
直接使用精确查询:找到city字段含有”上海“的文档;(注意理解ES中文档的概念)
聚合city字段:对city字段进行聚合,得到“上海”“北京”“深圳”的聚合结果,注意,只是聚合结果,不是具体的文档数据。此外我可以对聚合结果进行求和,求平均值等操作。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Elasticsearch DSL语法是一种用于构建Elasticsearch查询的Python库。它提供了一种简单而强大的方式来构建复杂的查询和聚合操作。DSL语法使用Python的面向对象语法来构建查询,使得代码易于阅读和维护。它支持各种查询类型,包括全文搜索、范围查询、过滤器、聚合等。DSL语法还提供了一些方便的方法来处理查询结果,如分页、排序、高亮等。总之,Elasticsearch DSL语法是一个非常有用的工具,可以帮助开发人员更轻松地构建和执行Elasticsearch查询。 ### 回答2: Elasticsearch DSL是一个基于Python的模块,它允许用户以更加方便的方式与Elasticsearch交互。DSL提供了一种更加简洁和可读的语法,减少了编写Elasticsearch查询的复杂性。DSL语法旨在尽可能地呈现Elasticsearch查询的结构。 DSL的主要语法包括以下几个方面: 1.索引:在DSL中,用户需要指定要查询的索引。例如,要查询名为“movies”的索引,用户应该使用以下语法:Index('movies')。 2.查询:设置检索的查询条件。在DSL中,用户可以使用各种查询类型来设置这些条件,例如term,match,range等。例如,要查询字段“title”等于“The Godfather”的文档,用户可以使用以下代码: from elasticsearch_dsl import Search from elasticsearch_dsl.query import Term s = Search().filter(Term(title='The Godfather')) 3.聚合:查询结果的聚合是DSL可以处理的另一个重要方面。用户可以使用各种聚合类型来获得有关查询结果的统计信息,例如总数,平均值,最大值等。例如,要统计字段“rating”的平均值,用户可以使用以下代码: from elasticsearch_dsl import Search from elasticsearch_dsl.aggs import Avg s = Search().agg(Avg('avg_rating', field='rating')) 4.排序:DSL中的排序允许用户根据指定的字段对查询结果进行排序。用户可以使用“sort”方法来设置排序规则,如以下代码所示: from elasticsearch_dsl import Search s = Search().sort('rating') 总之,Elasticsearch DSL语法提供了更强大的、更方便的方式与Elasticsearch进行交互。无论是搜索、聚合还是排序,DSL都可以帮助用户更加简单地构建复杂的查询。 ### 回答3: Elasticsearch DSL(Domain-Specific Language)是 Elasticsearch Python 客户端库实现的一种流畅的Python语法查询语句。Elasticsearch DSL 使Python开发人员能够以简单、易读的方式与 Elasticsearch 进行交互,更加方便地构建复杂的查询语句和聚合操作。下面我将从 dsl 查询语句的特点、基本语法和示例等方面进行详细说明。 Elasticsearch DSL查询语句的特点: 1.以Python方式编写查询语法,更加清晰易读; 2.类似于SQL语句的查询结构,更容易学习和使用; 3.高效的性能和精确的结果,提高了开发效率和用户体验; 4.可以轻松地与Python中的其他库进行集成,增加了开发的灵活性和可扩展性。 Elasticsearch DSL查询语句的基本语法: 1.创建一个查询: from elasticsearch_dsl import Search client = Elasticsearch() # 创建 Elasticsearch 客户端对象 search = Search(using=client) 2.匹配所有文档: search = Search(using=client, index="index_name").query("match_all") 3.匹配查询: search = Search(using=client, index="index_name").query("match", field="value") 4.范围查询: search = Search(using=client, index="index_name").query("range", field={"gte": 10, "lte":20}) 5.多个条件查询: search = Search(using=client, index="index_name").query("bool", must=[Q("match", title="python"), Q("match", content="elasticsearch")]) 6.结果排序和分页查询: search = Search(using=client, index="index_name").query("match", title="python").sort("-date").[0:10] Elasticsearch DSL查询语句示例: 1.查询内容为"python"的文章列表,并按发布时间倒序排序进行分页: s = Search().query("match", title="python").sort("-date").[0:10] response = s.execute() for hit in response: print(hit.title) 2.将通配符查询添加到筛选器列表中: search = Search().filter("wildcard", title="p*").query("match", body="python") response = search.execute() print("Total hits: ", response.hits.total.value) 3.使用子查询搜索不同版本的一个索引: s = Search(index="my-index").query( "bool", must_not=[Q("match", title="python")], should=[Q("match", title="java"), Q("match", title="ruby")] )执行 response = s.execute() print(response.hits.total.value) 总结:Elasticsearch DSL提供了Python化的语法来方便用户进行 Elasticsearch 操作。通过简单的语句即可实现复杂的查询,提高了开发效率和用户体验,开发人员可以很快学会并使用 Elasticsearch DSL

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值