1.term&terms查询
1.1 term查询
term查询代表完全匹配,搜索之前不会对你搜索的关键字进行分词,对你的关键字去文档分词库中去匹配内容
Kibana中实现
#term查询POST /sms-logs-index/sms-logs-type/_search{"from": 0,"size": 5,"query": {"term": {"province": {"value": "北京"}}}}
Java中实现
public class demo4 {RestHighLevelClient client = ESClient.getClient();String index = "sms-logs-index";String type = "sms-logs-type";@Testpublic void termQuery() throws Exception{//1.创建Request对象SearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.from(0);builder.size(5);builder.query(QueryBuilders.termQuery("province","北京"));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.获取到_source中的数据,并展示for (SearchHit hit : resp.getHits().getHits()) {Map<String, Object> result = hit.getSourceAsMap();System.out.println(result);}}}
1.2 terms查询
terms查询和term查询的机制一样,都不会将指定的查询关键字进行分词,直接去分词库中匹配,找到相应文档内容
terms是针对一个字段包含多个值的时候使用
term:where province = ‘北京’;
terms:where province = ‘北京’ or province = ? or province = ?
Kibana中实现**
#terms查询POST /sms-logs-index/sms-logs-type/_search{"query": {"terms": {"province": ["北京","山西","武汉"]}}}
Java中实现
@Testpublic void termsQuery() throws Exception{//1.创建Request对象SearchRequest request = new SearchRequest(index);request.types(type);//2.封装查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.termsQuery("province","北京","山西","武汉"));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出_sourcefor (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
2.match查询
match查询属于高层查询,他会根据你查询的字段类型不一样,采用不同的查询方式
- 查询的是日期或者是数值的话,他会基于你的字符串查询内容转换为日期或者数值对待
- 如果查询的内容是一个不能被分词的内容(keyword),match查询不会对你的查询关键字进行分词
- 如果查询的内容是一个可以被分词的内容(text),match会将你指定的查询内容根据一定的方式去分词,去分词库中匹配指定的内容
match查询,实际底层就是多个term查询,将多个term查询的结果给你封装到了一起
2.1 match_all查询
查询全部内容,不指定任何查询条件
Kibana中实现
#match_all查询POST /sms-logs-index/sms-logs-type/_search{"query": {"match_all": {}}}
Java中实现
@Testpublic void matchAllQuery() throws Exception{//1.创建RequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.matchAllQuery());builder.size(20); //es默认只查询10条数据request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}System.out.println(resp.getHits().getHits().length);}
2.2 match查询
指定一个Field作为查询的条件
Kibana中实现
#match查询POST /sms-logs-index/sms-logs-type/_search{"query": {"match": {"smsContent": "收货安装"}}}
Java中实现
@Testpublic void matchQuery() throws Exception{//1.创建RequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.matchQuery("smsContent","收货安装"));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}System.out.println(resp.getHits().getHits().length);}
2.3 布尔match查询
基于一个Field匹配的内容,采用and或者or的方式连接
Kibana中实现
#布尔match查询POST /sms-logs-index/sms-logs-type/_search{"query": {"match": {"smsContent": {"query": "中国 健康","operator": "and"}}}}#布尔match查询POST /sms-logs-index/sms-logs-type/_search{"query": {"match": {"smsContent": {"query": "中国 健康","operator": "or"}}}}
Java中实现
@Testpublic void booleanMatchQuery() throws Exception{//1.创建RequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.matchQuery("smsContent","中国 健康").operator(Operator.OR));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
2.4 multi_match查询
match针对一个field做检索,multi_match针对多个field进行检索,多个field对应一个text
Kibana中实现
#multi_match查询POST /sms-logs-index/sms-logs-type/_search{"query": {"multi_match": {"query": "北京","fields": ["province","smsContent"]}}}
Java中实现
@Testpublic void multiMatchQuery() throws Exception{//1.创建RequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.multiMatchQuery("北京","province","smsContent"));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
3.其他查询
3.1 id查询
根据id查询where id=?
Kibana中实现
#id查询GET /sms-logs-index/sms-logs-type/21
Java中实现
@Testpublic void findById() throws Exception{//1.创建GetRequestGetRequest request = new GetRequest(index, type, "21");//2.执行查询GetResponse resp = client.get(request, RequestOptions.DEFAULT);//3.输出结果System.out.println(resp.getSourceAsMap());}
3.2 ids查询
Kibana中实现
#ids查询POST /sms-logs-index/sms-logs-type/_search{"query": {"ids": {"values": ["21","22","23"]}}}
Java中实现
@Testpublic void findByIds() throws IOException {//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.idsQuery().addIds("21","22","23"));request.source(builder);//3.执行SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
3.3 prefix查询
前缀查询,可以通过一个关键字去指定一个Field的前缀,从而查询到指定的文档
Kibana中实现
#prefix查询POST /sms-logs-index/sms-logs-type/_search{"query": {"prefix": {"corpName": {"value": "途虎"}}}}
Java中实现
@Testpublic void findByPrefix() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.prefixQuery("corpName","盒马"));request.source(builder);//3.执行SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
3.4 fuzzy查询
模糊查询,我们输入字符的大概,ES就可以去根据输入的内容大概去匹配一下结果
Kibana中实现
#fuzzy查询POST /sms-logs-index/sms-logs-type/_search{"query": {"fuzzy": {"corpName": {"value": "盒马先生","prefix_length": 2 #指定前面几个字符是不允许出现错误的}}}}
Java中实现
@Testpublic void findByFuzzy() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.fuzzyQuery("corpName","盒马先生").prefixLength(2));request.source(builder);//3.执行SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
3.5 wildcard查询
通配查询,和MySQL中的like是一个套路,可以在查询时,在字符串中指定通配符和占位符?
*Kibana中实现
#wildward查询POST /sms-logs-index/sms-logs-type/_search{"query": {"wildcard": {"corpName": {"value": "中国*" #可以使用*和? 指定通配符和占位符}}}}
Java中实现
@Testpublic void findByWildCard() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.wildcardQuery("corpName","中国*"));request.source(builder);//3.执行SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
3.6 range查询
范围查询,只针对数值类型,对某一个Field进行大于或者小于的范围指定
Kibana中实现
#range查询POST /sms-logs-index/sms-logs-type/_search{"query": {"range": {"fee": {"gt": 5,"lte": 10}}}}
Java中实现
@Testpublic void findByRange() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.rangeQuery("fee").gt(5).lte(10));request.source(builder);//3.执行SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
3.7 regexp查询
正则规则,通过你编写的正则表达式区匹配内容
Kibana中实现
#regexp查询POST /sms-logs-index/sms-logs-type/_search{"query": {"regexp": {"mobile": "180[0-9]{8}"}}}
Java中实现
@Testpublic void findByRange() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.rangeQuery("fee").gt(5).lte(10));request.source(builder);//3.执行SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
PS:**prefix查询,fuzzy查询,wildcard查询和regexp查询**效率相对比较低,要求效率比较高时,避免去使用
**
4.深分页Scroll
4.1 深分页原理
from+size分页方式缺点:
- Elasticsearch对于from+size是有限制的,from和size两者只和不能超过1w
from+size方式与scroll+size方式的原理:
- from+size方式在Elasticsearch查询数据的方式:
- 1.将用户指定的内容进行分词(是否会分词由查询方式与指定内容决定)
- 2.将词汇拿去词库中进行检索,得到多个文档的id
- 3.去各个分片中拉去指定的数据,耗时较长
- 4.将数据根据score进行排序,耗时较长
- 5.根据from的值,将查询到的数据舍弃一部分
- 6.返回结果
- scroll+size方式在Elasticsearch查询数据的方式:
- 1.将用户指定的内容进行分词(是否会分词由查询方式与指定内容决定)
- 2.将词汇拿去词库中进行检索,得到多个文档的id
- 3.将文档的id存放在一个Elasticsearch的上下文(内存)中
- 4.根据你指定的size去Elasticsearch中检索指定个数的数据,数据被拿到之后会从上下文(内存)中删除
- 5.如果需要下一页数据,直接去Elasticsearch的上下文(内存)中,找后续内容
- 6.循环第4,5步
4.2 深分页使用
Kibana中实现
#执行scroll查询,返回第一页数据,并且将文档id信息存放在ES上下文中,指定生存时间1mPOST /sms-logs-index/sms-logs-type/_search?scroll=5m{"query": {"match_all": {}},"size": 2,"sort": [{"fee": {"order": "desc"}}]}#根据scroll查询下一页的数据POST /_search/scroll{#scroll_id由上面的scroll查询生成"scroll_id":"DnF1ZXJ5VGhlbkZldGNoAwAAAAAAAAfcFnZvMTh2N1lzVHJtUmxhUkZ2aWY3T1EAAAAAAAAH3RZ2bzE4djdZc1RybVJsYVJGdmlmN09RAAAAAAAAB94Wdm8xOHY3WXNUcm1SbGFSRnZpZjdPUQ==","scroll":"5m"}#删除scroll在ES上下文中的数据DELETE /_search/scroll/DnF1ZXJ5VGhlbkZldGNoAwAAAAAAAAfcFnZvMTh2N1lzVHJtUmxhUkZ2aWY3T1EAAAAAAAAH3RZ2bzE4djdZc1RybVJsYVJGdmlmN09RAAAAAAAAB94Wdm8xOHY3WXNUcm1SbGFSRnZpZjdPUQ==
Java中实现
@Testpublic void scrollQuery() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定scroll信息request.scroll(TimeValue.timeValueMinutes(1L));//3.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();builder.size(2);builder.sort("fee", SortOrder.DESC);builder.query(QueryBuilders.matchAllQuery());request.source(builder);//4.获取返回结果scrollId,sourceSearchResponse resp = client.search(request, RequestOptions.DEFAULT);String scrollId = resp.getScrollId();System.out.println("--------------首页--------------");for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}while (true){//5.循环 - 创建SearchScrollRequestSearchScrollRequest scrollRequest = new SearchScrollRequest(scrollId);//6.指定scrollId的生存时间scrollRequest.scroll(TimeValue.timeValueMinutes(1L));//7.执行查询获取返回结果SearchResponse searchResponse = client.scroll(scrollRequest, RequestOptions.DEFAULT);//8.判断是否查询到了数据,输出SearchHit[] hits = searchResponse.getHits().getHits();if (hits != null && hits.length > 0){System.out.println("--------------下一页--------------");for (SearchHit hit : hits) {System.out.println(hit.getSourceAsMap());}}else{//9.没有查询到数据,跳出循环System.out.println("--------------结束--------------");break;}}//10.创建ClearScrollRequestClearScrollRequest clearScrollRequest = new ClearScrollRequest();//11.指定ScrollIdclearScrollRequest.addScrollId(scrollId);//12.删除ScrollIdClearScrollResponse clearScrollResponse = client.clearScroll(clearScrollRequest, RequestOptions.DEFAULT);//13.输出结果System.out.println("删除scroll:" + clearScrollResponse.isSucceeded());}
5.delete-by-query
根据term,match等方式去删除大量的文档。
Kibana中实现
#delete-by-queryPOST /sms-logs-index/sms-logs-type/_delete_by_query{"query": {"range": {"fee": {"lt": 4}}}}
Java中实现
@Testpublic void deleteByQuery() throws IOException {//1.创建DeleteByQueryRequestDeleteByQueryRequest request = new DeleteByQueryRequest(index);request.types(type);//2.指定检索的条件,和SearchRequest指定Query的方式不一样request.setQuery(QueryBuilders.rangeQuery("fee").lte(4));//3.执行删除BulkByScrollResponse resp = client.deleteByQuery(request, RequestOptions.DEFAULT);//4.输出返回结果System.out.println(resp.toString());}
如果你需要删除的内容,是index下的大部分数据,推荐创建一个全新的index,将保留的文档内容,添加到新的文档。
6.复合查询
6.1 bool查询
复合过滤器,将你的多个查询条件,以一定的逻辑组合在一起
- must:将所有的条件,用must组合在一起,表示And的意思
- must_not:将must_not中的条件,全部都不能匹配,表示Not的意思
- should:所有的条件,用should组合在一起,表示Or的意思
Kibana中实现
#查询省份为武汉或者北京#运营商不是联通#smsContent中包含中国和平安#bool查询POST /sms-logs-index/sms-logs-type/_search{"query": {"bool": {"should": [{"term": {"province": {"value": "北京"}}},{"term": {"province": {"value": "武汉"}}}],"must_not": [{"term": {"operatorId": {"value": "2"}}}],"must": [{"match": {"smsContent": "中国"}},{"match": {"smsContent": "平安"}}]}}}
Java中实现
@Testpublic void BoolQuery() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();//查询省份为武汉或者北京boolQueryBuilder.should(QueryBuilders.termQuery("province","武汉"));boolQueryBuilder.should(QueryBuilders.termQuery("province","北京"));//运营商不是联通boolQueryBuilder.mustNot(QueryBuilders.termQuery("operatorId","2"));//smsContent中包含中国和平安boolQueryBuilder.must(QueryBuilders.matchQuery("smsContent","中国"));boolQueryBuilder.must(QueryBuilders.matchQuery("smsContent","平安"));builder.query(boolQueryBuilder);request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
6.2 boosting查询
boosting查询可以帮助我们去影响查询后的score
- positive:只有匹配上positive的查询内容,才会被放到返回的结果集中
- negative:如果匹配上positive并且也匹配上了negative,就可以降低这样的文档score
- negative_boost:指定系数,必须小于1.0
关于查询时,分数是如何计算的:
- 搜索的关键字在文档中出现的频次越高,分数就越高
- 指定的文档内容越短,分数就越高
- 在搜索时,指定的关键字也会被分词,这个被分词的内容,被分词库匹配的个数越多,分数越高
Kibana中实现
#boosting查询 收货安装POST /sms-logs-index/sms-logs-type/_search{"query": {"boosting": {"positive": {"match": {"smsContent": "收货安装"}},"negative": {"match": {"smsContent": "王五"}},"negative_boost": 0.5}}}
Java中实现
@Testpublic void BoostingQuery() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定查询条件SearchSourceBuilder builder = new SearchSourceBuilder();BoostingQueryBuilder boostingQueryBuilder = QueryBuilders.boostingQuery(QueryBuilders.matchQuery("smsContent", "收货安装"),QueryBuilders.matchQuery("smsContent", "王五")).negativeBoost(0.5f);builder.query(boostingQueryBuilder);request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
7.filter查询
query查询,根据你的查询条件,去计算文档的匹配度得到一个分数,并且根据分数进行排序,不会做缓存的
filter,根据你的查询条件去查询文档,不去计算分数,而且filter会对经常被过滤的数据进行缓存
Kibana中实现
#filter查询POST /sms-logs-index/sms-logs-type/_search{"query": {"bool": {"filter": [{"term":{"corpName": "盒马鲜生"}},{"range": {"fee": {"lte": 5}}}]}}}
Java中实现
@Testpublic void filter() throws Exception{//1.SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.查询条件SearchSourceBuilder builder = new SearchSourceBuilder();BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();boolQueryBuilder.filter(QueryBuilders.termQuery("corpName","盒马鲜生"));boolQueryBuilder.filter(QueryBuilders.rangeQuery("fee").lte(5));builder.query(boolQueryBuilder);request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
8.高亮查询
高亮查询就是用户输入的关键字,以一定的特殊样式展示给用户,让用户知道为什么这个结果被检索出来
高亮展示的数据,本身就是文档中的一个Field,单独将Field以highlight的形式返回给你
Elasticsearch提供了一个highlight属性,和query同级别的
- fragment_size:指定高亮数据展示多少个字符回来
- pre_tags:指定前缀标签,举个例子
- post_tags:指定后缀标签,举个例子
- fields:指定哪几个Field以高亮的形式返回
Kibana中实现
#highlight查询POST /sms-logs-index/sms-logs-type/_search{"query": {"match": {"smsContent": "盒马"}},"highlight": {"fields": {"smsContent": {}},"pre_tags": "<font color='red'>","post_tags": "</font>","fragment_size": 10}}
Java中实现
@Testpublic void highLightQuery() throws Exception{//1.SearchRequestSearchRequest request = new SearchRequest(index);//2.指定查询条件(高亮)SearchSourceBuilder builder = new SearchSourceBuilder();builder.query(QueryBuilders.matchQuery("smsContent","盒马"));HighlightBuilder highlightBuilder = new HighlightBuilder();highlightBuilder.field("smsContent",10).preTags("<font color='red'>").postTags("</font>");builder.highlighter(highlightBuilder);request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.高亮数据,输出for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit);}}
9.聚合查询
Elasticsearch聚合查询寻和MySQL的聚合查询类似,Elasticsearch的聚合查询比MySQL要强大得多,Elasticsearch提供的统计数据的方式多种多样
#ES聚合查询的RESTful语法POST /index/type/_search{"aggs": {"名字(agg)": {"agg_type": {"属性": "值"}}}}
9.1 去重计数查询
去重计数查询,即Cardinality,第一步先将返回的文档中的一个指定的field进行去重,统计一共有多少条
Kibana中实现
#去重计数查询 北京 上海 武汉 山西POST /sms-logs-index/sms-log-type/_search{"aggs": {"agg": {"cardinality": {"field": "province"}}}}
Java中实现
@Testpublic void cardinality() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定使用的聚合查询方式SearchSourceBuilder builder = new SearchSourceBuilder();builder.aggregation(AggregationBuilders.cardinality("agg").field("province"));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.获取返回结果Cardinality agg = resp.getAggregations().get("agg");long value = agg.getValue();System.out.println(value);}
9.2 范围统计
统计一定范围内出现的文档的个数,比如,针对某个Field的值在0~100,100~200,200~300之间文档出现的个数分别是多少。范围统计可以针对普通的数值,这对时间类型,针对ip类型都可以做相应的统计。
range,date_range,ip_range
Kibana中实现
#数值方式范围统计POST /sms-logs-index/sms-logs-type/_search{"aggs": {"agg": {"range": {"field": "fee","ranges": [{"to": 5},{"from": 5,"to": 10},{"from": 10}]}}}}#时间方式范围统计POST /sms-logs-index/sms-logs-type/_search{"aggs": {"agg": {"date_range": {"field": "createDate","format": "yyyy","ranges": [{"to": 2000},{"from": 2000}]}}}}#ip统计方式POST /sms-logs-index/sms-logs-type/_search{"aggs": {"agg": {"ip_range": {"field": "ipAddr","ranges": [{"to": "10.126.2.9"},{"from": "10.126.2.9"}]}}}}
Java中实现
@Testpublic void range() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定使用的聚合查询方式SearchSourceBuilder builder = new SearchSourceBuilder();builder.aggregation(AggregationBuilders.range("agg").field("fee").addUnboundedTo(5).addRange(5,10).addUnboundedFrom(10));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.获取返回结果Range agg = resp.getAggregations().get("agg");for (Range.Bucket bucket : agg.getBuckets()) {String key = bucket.getKeyAsString();Object from = bucket.getFrom();Object to = bucket.getTo();long docCount = bucket.getDocCount();System.out.println(String.format("key: %s,from: %s,to: %s,docCount: %s",key,from,to,docCount));}}
9.3 统计聚合查询
可以查询指定Field的最大值,最小值,平均值,平方和等
使用extended_stats
Kibana中实现
#统计聚合查询POST /sms-logs-index/sms-logs-type/_search{"aggs": {"agg": {"extended_stats": {"field": "fee"}}}}
Java中实现
@Testpublic void extendsStats() throws Exception{//1.创建SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2.指定使用的聚合查询方式SearchSourceBuilder builder = new SearchSourceBuilder();builder.aggregation(AggregationBuilders.extendedStats("agg").field("fee"));request.source(builder);//3.执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4.获取返回结果ExtendedStats agg = resp.getAggregations().get("agg");double max = agg.getMax();double min = agg.getMin();System.out.println("fee的最大值为:" + max + ",最小值为:" + min);}
10.地图经纬度查询
Elasticsearch中提供了一个数据类型geo_point,这个类型就是从来存储经纬度的
创建一个带geo_point类型的索引,并添加测试数据
#创建一个索引,指定一个name,locationPUT /map{"settings": {"number_of_shards": 5,"number_of_replicas": 1},"mappings": {"map": {"properties": {"name": {"type": "text"},"location": {"type": "geo_point"}}}}}#添加测试数据PUT /map/map/1{"name": "天安门","location": {"lon": 116.403981,"lat":39.914492}}PUT /map/map/2{"name": "海淀公园","location": {"lon": 116.302509,"lat": 39.991152}}PUT /map/map/3{"name": "北京动物园","location": {"lon": 116.343184,"lat": 39.947468}}
10.1 Elasticsearch的地图检索方式
geo_distance:直线距离检索方式
geo_bounding_box:以两个点确定一个矩形,获取在矩形内的全部数据
geo_polygon:以多个点,确定一个多边形,获取多边形内的全部数据
10.2 基于RESTful实现地图检索
Kibana中实现
#geo_distancePOST /map/map/_search{"query": {"geo_distance": {"location": { #确定一个点"lon": 116.433733,"lat": 39.909404},"distance": 3000, #确定半径"distance_type": "arc" #指定形状为圆形}}}#geo_bounding_boxPOST /map/map/_search{"query": {"geo_bounding_box": {"location": {"top_left": { #左上角的坐标点"lon": 116.326943,"lat": 39.954990},"bottom_right": { #右下角的坐标点"lon": 116.433446,"lat": 39.908737}}}}}#geo_polygonPOST /map/map/_search{"query": {"geo_polygon": {"location": {"points": [ #指定多个点确定一个多边形{"lon": 116.298916,"lat": 39.99878},{"lon": 116.29561,"lat": 39.972576},{"lon": 116.327661,"lat": 39.984739}]}}}}
Java中实现
@Testpublic void geoPolygon() throws IOException {//1. SearchRequestSearchRequest request = new SearchRequest(index);request.types(type);//2. 指定检索方式SearchSourceBuilder builder = new SearchSourceBuilder();List<GeoPoint> points = new ArrayList<>();points.add(new GeoPoint(39.99878,116.298916));points.add(new GeoPoint(39.972576,116.29561));points.add(new GeoPoint(39.984739,116.327661));builder.query(QueryBuilders.geoPolygonQuery("location",points));request.source(builder);//3. 执行查询SearchResponse resp = client.search(request, RequestOptions.DEFAULT);//4. 输出结果for (SearchHit hit : resp.getHits().getHits()) {System.out.println(hit.getSourceAsMap());}}
