SpringCloud(9)— Elasticsearch聚合和自动补全

news/2024/12/29 0:56:14/

SpringCloud(9)— Elasticsearch聚合和自动补全

一 数据聚合

1.聚合的分类

聚合(aggregations)可以实现对文档数据的统计,分析,运算。常见的聚合有三种:

1.桶聚合(Bucket)

text 不支持 桶聚合

桶聚合(Bucket)用来对文档做分组,其中比较常见的有:

  • TermAggregation:按照文档的字段值进行分组,类似 MySql 中的 group by
  • Date Histogram:按照日期阶梯分组,例如 一周或者一月 为一组

2.度量聚合(Metric)

text 和 keyword 不支持 度量聚合

度量聚合用于计算一些值,比如最大值,最小值,平均值等。常见的有:

  • Avg:平均值
  • Max:最大值
  • Min:最小值
  • Sum:求和
  • Stats:同时求 Avg, Max, Min, Sum 等

3.管道聚合(pipeline)

管道聚合以其他聚合的结果为基础做聚合

参与聚合的字段类型,不能是 text 类型

一般为 keyword,date,bool,integer 等

2.DSL实现Bucket聚合

以下是语法示例:

GET /【indexName】/_search
{"size":"分页值,默认为10,只做聚合不做数据分页时设置为0,则只会返回聚合结果而不会返回文档","aggs": {"聚合名称,可自定义": {"聚合类型,一般为 terms": {"field": "字段名","size": 返回的数据量}}}
}

默认情况下,Bucket 聚合会统计文档数量记为 _count,且按照 _count 进行倒序排序。

如果需要修改的话,则只需要增加 order 属性,并设置排序规则即可

以下是以 brand 为例的聚合示例:

GET /hotel/_search
{"size":0,"aggs": {"brandAgg": {"terms": {"field": "brand","size": 20,"order":{"_count":"asc"}}}}
}

在这里插入图片描述

默认情况下,Bucket 聚合是对索引库的所有文档做聚合,对内存的消耗非常大。

我们可以通过增加 query 限定聚合的文档范围。

例如,只对 价格(price) 在 200-300 范围内的数据做聚合:

GET /hotel/_search
{"query": {"range": {"price": {"gte": 200,"lte": 300}}}, "size":0,"aggs": {"brandAgg": {"terms": {"field": "brand","size": 20,"order":{"_count":"asc"}}}}
}

3.DSL实现 Metrics 聚合

利用 Stats 聚合,获取指定字段的各项度量值

GET /hotel/_search
{"size":0,"aggs": {"brandAgg": {"terms": {"field": "brand","size": 20},"aggs":{           //是 brandAgg 聚合后的子聚合,也就是分组后对每组分别计算"score_stats":{  //聚合名称"stats": {     //聚合类型,此处可以是 min,max,avg等"field": "score"  //聚合的字段值,只能是数值类型,因为只有数组可以进行加减乘除}}}}}
}

DSL 示例:

GET /hotel/_search
{"size":0,"aggs": {"brandAgg": {"terms": {"field": "brand","size": 20},"aggs":{"score_stats":{"stats": {"field": "score"}}}}}
}

运行 DSL 代码,得到下图右侧的结果,利用 stats 同时计算出了最大值,最小值,和值,平均值以及数量。

在这里插入图片描述

此时如果想要对按照聚合之后的值排序,则应当使用 score_stats 中的属性来定义。

例如,按照 score 的最大值进行排序

GET /hotel/_search
{"size":0,"aggs": {"brandAgg": {"terms": {"field": "brand","size": 20,"order": {"score_stats.max": "desc"}},"aggs":{"score_stats":{"stats": {"field": "score"}}}}}
}

4.RestClient实现聚合

先参考以下对照图

在这里插入图片描述

以下是测试示例:

@Test
public void testAggregationBrand() throws IOException {//1.创建 SearchRequest 对象,指定索引库名称SearchRequest request = new SearchRequest("hotel");//2.去掉文档数据request.source().size(0);request.source().aggregation(AggregationBuilders//设置聚合类型为 term,且为聚合起名.terms("brandAgg")//设置需要聚合的字段.field("brand")//设置返回的数据量.size(20)//设置排序,.order(BucketOrder.aggregation("_count",true)));//3.发送请求SearchResponse response = restHighLevelClient.search(request, RequestOptions.DEFAULT);//获取全部聚合结果Aggregations aggregations = response.getAggregations();//根据 聚合名称 获取聚合结果Terms brandTerms = aggregations.get("brandAgg");//获取桶List<? extends Terms.Bucket> buckets = brandTerms.getBuckets();//遍历数据for (Terms.Bucket bucket : buckets) {String brandName = bucket.getKeyAsString();Long docCount = bucket.getDocCount();System.out.println(brandName+","+docCount);}
}

需要注意:使用 AggregationBuilders 来构建一个 aggregation 对象

二 自动补全

1.拼音分词器的使用

GitHub地址:elasticsearch-analysis-pinyin

安装步骤:

  1. 下载指定版本(与 es 版本保持一致,文档使用 v7.12.1)的 elasticsearch-analysis-pinyin
  2. 解压并上传至 es 容器挂载插件的目录(与 ik分词器 同一个目录)
  3. 重启 es 容器
  4. 测试
GET /_analyze
{"text":["如家"],"analyzer": "pinyin"
}

返回一下结果说明分词器安装成功,且成功分词

在这里插入图片描述

2.自定义分词器

1.分词器的构成

  • character filters:在 tokenizer 之前对文本进行处理,例如 删除字符,替换字符等
  • tokenizer:将文本按照一定的规则切割成词条(term),例如 keyword。
  • tokenizer filter:将 tokenizer 输出的词条做进一步的处理,例如大小写转换,同义词处理,拼音处理等

在这里插入图片描述

自定义分词器时不一定三部分都需要。根据实际业务需求即可。例如以下示例,只有 tokenizer 和 filter两部分,并没有 character

2.自定义分词器的实现

在创建索引库时,通过 settings 来配置自定义的 analyzer(分词器)

因为自定义分词器是在创建索引库时指定,所以自定义分词器只针对当前的索引库生效

以下是语法示例:

PUT /【indexName】
{"settings": {"analysis": {"analyzer": {"自定义分词器的名称":{"tokenizer":"分词器名称","filter":"分词器名称"}}}}
}

实现示例:

PUT /test
{"settings": {"analysis": {"analyzer": {"my_analyzer":{"tokenizer":"ik_max_word","filter":"pinyin"}}}}
}

以上这种默认实现会将汉字分为一个一个的拼音,这并非我们想要的。参考文档,设置其他相关参数。

以下是实现示例:

PUT /test
{"settings": {"analysis": {"analyzer": {"my_analyzer":{"tokenizer":"ik_max_word","filter":"py"}},"filter": {"py":{"type":"pinyin",   // 使用拼音分词器,以下为拼音分词器的部分参数设置"keep_full_pinyin":false,"keep_joined_full_pinyin":true,"keep_original":true,"limit_first_letter_length":16,"remove_duplicated_term":true,"none_chinese_pinyin_tokenize":false}}}}
}
  • my_analyzer:自定义的分词器名称
  • py:自定义的分词器名称

3.拼音分词器注意事项

拼音分词器是和在创建倒排索引时使用,但不能在搜索时使用(会搜到同音词),因此字段在创建索引时应该使用创建的分词器,字段在搜索时应该使用 ik_smart 分词器

PUT /test
{"settings": {"analysis": {"analyzer": {"my_analyzer":{"tokenizer":"ik_max_word","filter":"py"}},"filter": {"py":{"type":"pinyin","keep_full_pinyin":false,"keep_joined_full_pinyin":true,"keep_original":true,"limit_first_letter_length":16,"remove_duplicated_term":true,"none_chinese_pinyin_tokenize":false}}}},"mappings": {"properties": {"name":{"type": "text","analyzer": "my_analyzer","search_analyzer": "ik_smart"}}}
}

3.自动补全查询

1.Completion Suggester

elasticsearch 提供了 Completion Suggester 查询来实现自动补全,这个查询会匹配以用户输入的内容开头的词条并返回。

为了提高查询提高效率,需要堆文档中的字段做一些约束:

  • 要求查询字段必须为 completion 类型
  • 字段的内容一般是用来补全的多个词条形成的数据

在这里插入图片描述

2.语法示例

GET /test/_search
{"suggest": {"自定义suggest名称": {"text": "YOUR TEXT","completion":{"field":"字段名",        // 补全查询的字段"skip_duplicates":true, //跳过重复的"size":10               // 获取前10条结果}}}
}

三 实现搜索的自动补全

1.修改原有的数据结构

使用自定义分词器,并且增加 suggestion 字段,用于实现自动补全

PUT /hotel
{"settings": {"analysis": {"analyzer": {"text_analyzer":{"tokenizer":"ik_max_word","filter":"py"},"completion_analyzer":{"tokenizer":"keyword","filter":"py"}},"filter": {"py":{"type":"pinyin","keep_full_pinyin":false,"keep_joined_full_pinyin":true,"keep_original":true,"limit_first_letter_length":16,"remove_duplicated_term":true,"none_chinese_pinyin_tokenize":false}}}}, "mappings": {"properties": {"suggestion":{"type": "completion","analyzer": "completion_analyzer"},"all":{"type": "text","analyzer": "text_analyzer","search_analyzer": "ik_smart"},"id":{"type": "keyword"},"name":{"type": "text","copy_to": "all", "analyzer": "text_analyzer","search_analyzer": "ik_smart"},"address":{"type": "keyword","index": false},"price":{"type": "double"},"score":{"type": "integer"},"brand":{"type": "keyword","copy_to": "all"},"city":{"type": "keyword","copy_to": "all"},"starName":{"type": "keyword","copy_to": "all"},"business":{"type": "keyword","copy_to": "all"},"location":{"type": "geo_point"},"pic":{"type": "keyword","index": false},"isAD":{"type": "boolean"}}}
} 

2.重新导入数据

修改实体类与文档的对应关系,重新导入数据

@Data
@NoArgsConstructor
public class HotelDoc {private Long id;private String name;private String address;private Integer price;private Integer score;private String brand;private String city;private String starName;private String business;private String location;private String pic;private Object distance;/*** 新增自动补全字段*/private List<String> suggestion;/*** 广告*/public Boolean isAD;public HotelDoc(Hotel hotel) {this.id = hotel.getId();this.name = hotel.getName();this.address = hotel.getAddress();this.price = hotel.getPrice();this.score = hotel.getScore();this.brand = hotel.getBrand();this.city = hotel.getCity();this.starName = hotel.getStarName();this.business = hotel.getBusiness();this.location = hotel.getLatitude() + "," + hotel.getLongitude();this.pic = hotel.getPic();//处理广告字段(isAD)格式if (hotel.getIsAD() == 1) {this.isAD = true;} else {this.isAD = false;}//处理自动补全信息List<String> list =new ArrayList<>();list.add(hotel.getName());list.add(hotel.getBrand());//商圈信息含有"/"时视为多个,做切割处理String business = hotel.getBusiness();if (business.contains("/")) {String[] businessArr = business.split("/");Collections.addAll(list,businessArr);}this.setSuggestion(list);}
}

导入数据完成,使用 DSL 语句进行测试

GET /hotel/_search
{"suggest": {"textSuggestion": {"text": "s","completion": {"field": "suggestion","skip_duplicates":true,"size":10}}}
}

3.RestClient实现自动补全

先看格式对照

在这里插入图片描述

编写测试代码,实现自动补全

@Test
void testSuggestion() throws IOException {//1.创建 SearchRequest 对象SearchRequest searchRequest=new SearchRequest("hotel");//2.构造 DSL 语句searchRequest.source().suggest(new SuggestBuilder().addSuggestion("textSuggestion",SuggestBuilders.completionSuggestion("suggestion").prefix("bj").skipDuplicates(true).size(10)));//3.发送请求SearchResponse response = restHighLevelClient.search(searchRequest, RequestOptions.DEFAULT);//4.处理数据Suggest suggest = response.getSuggest();CompletionSuggestion suggestion= suggest.getSuggestion("textSuggestion");for (CompletionSuggestion.Entry.Option option : suggestion.getOptions()) {String text = option.getText().toString();System.out.println(text);}
}

解析结果实际按照Json格式去逐层解析:

在这里插入图片描述

四 数据同步

  1. elasticsearch中的数据来源于数据库,因此数据库发生改变时, es 中的数据必须也跟着变。这就是 es 和数据库之间的同步问题
  2. 在微服务项目中,不同的业务一般运行在不同的服务器上,该如何解决?

1.同步调用

优缺点:

实现简单,粗暴。但是业务耦合度高

在这里插入图片描述

2.异步调用

优缺点:

低耦合,实现难度一般。但是过于依赖MQ的可靠性

在这里插入图片描述

3.监听binlog

优缺点:

耦合度完全解除。但是开启之后mysql的负担增加,且实现复杂度高

在这里插入图片描述

2022-12-28,增加 第四节 数据同步
完结撒花!!!


http://www.ppmy.cn/news/7429.html

相关文章

Linux内存从0到1学习笔记(8.12 dma-buf导出器和导入器使用示例 三)

前面解释了使用dma_buf所涉及的几个重要的结构体,今天我们来介绍下dma_buf使用的具体示例。 mock_dmabuf_exp.c示例 #include <linux/dma-buf.h> #include <linux/module.h> #include <linux/slab.h> struct dma_buf *dmabuf_of_exporter; EXPORT_SYMBOL(d…

k8s docker空间使用分析与清理

目录 1. 前言 2. 常见问题场景 3. docker 磁盘空间使用分析及处理 4. 总结 1. 前言 使用 Docker 运行业务一段时间后&#xff0c;可能会出现宿主节点的磁盘容量占用高&#xff0c;导致宿主机磁盘空间不足等异常&#xff0c;对业务造成影响。本文对 Docker 的空间占用进行汇…

机器学习基石2(noise和error)

文章目录一、什么是noise&#xff1f;二、什么是error&#xff1f;三、常用error一、什么是noise&#xff1f; 我们之前的讨论都是一种理想化的说明&#xff0c;比如数据来源于目标函数fff&#xff0c;似乎我们手里拿到的数据是这样得来的&#xff0c;随机取一个输入&#xff0…

vulnhub DC系列 DC-1

目录 下载地址 漏洞分析 信息收集 漏洞尝试 漏洞利用 flag1 flag2 flag3 flag4 提权 下载地址 DC-1.zip (Size: 733 MB)Download: http://www.five86.com/downloads/DC-1.zipDownload (Mirror): https://download.vulnhub.com/dc/DC-1.zip使用方式:ova文件直接使用vm…

人文笔记(芒格人类误判心理学 + 韩信研究)

芒格人类误判心理学 1、奖励和惩罚&#xff0c;超级反应倾向 2、喜欢/热爱倾向 3、讨厌/憎恨倾向 4、避免怀疑倾向 5、避免不一致倾向 6、好奇心倾向 7、康德式公平倾向 8、艳羡/妒忌倾向 9、回馈倾向 10、受简单联想影响的倾向 11、简单的、避免痛苦的心理否认 12、自视甚高的…

知识付费海哥:知识变现三剑客

小时候看武侠戏&#xff0c;那些古代剑客&#xff0c;飞檐走壁&#xff0c;神出鬼没&#xff0c;行走江湖&#xff0c;行侠仗义&#xff0c;牛逼的很&#xff01;心里贼仰慕&#xff01; 记忆最深的&#xff0c;是剑客们稳、准、狠的绝技剑法&#xff0c;剑起头落&#xff0c;一…

2022圣诞树(C语言摇钱树版本)

逐梦编程&#xff0c;让中华屹立世界之巅。 简单的事情重复做,重复的事情用心做,用心的事情坚持做&#xff1b; 文章目录前言一、个人感悟二、圣诞树由来三、圣诞树发展历史演变四、常见的圣诞树种类五、摇钱圣诞树效果展示六、实现思路七、编码实现总结新壁纸前言 时光飞逝&a…

前端面试八股文(详细版)—中

上篇详细讲解了 HTML 、CSS 、JavaScript 、计算机网络知识等方面的内容&#xff0c;本文将详细讲解 React 、git 、webpack等内容&#xff0c;预祝各位成功上岸&#xff01; React框架 使用 react 脚手架搭建项目 npx create-react-app app-name class 和 function 的区别 …