当前位置: 首页 > news >正文

SpringBoot集成Elasticsearch实例

SpringBoot项目集成Elasticsearch实例

导包

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-elasticsearch</artifactId>
</dependency>

配置es连接

spring:data:elasticsearch:cluster-name: elasticsearchcluster-nodes: 127.0.0.1:9300

准备query接收查询条件

@Data
public class CarSearchQuery extends BaseQuery {private Long carType;private Double maxPrice;private Double minPrice;//0 以下 1 以上private Integer carAgeType;private Integer carAge;//是否超值private Integer costEffective;//急售private Integer rushSale;//准新车private Integer quasiNewCar;//可迁全国private Integer transitiveCountry;//排序字段private String sortField;//排序类型 desc降序 asc升序号private String sortType;private Double longitude;private Double latitude;private Double distance;private Long shopId;
}

准备Controller

@RestController
@RequestMapping("/car/search")
public class CarSearchController {@Autowiredprivate ICarSearchService carSearchService;@PostMappingpublic AjaxResult search(@RequestBody CarSearchQuery query) {PageList<CarDoc> search = carSearchService.search(query);return AjaxResult.me().setData(search);}
}

定义Document文档类型

@Data
@AllArgsConstructor
@NoArgsConstructor
@Document(indexName = "example-car", type = "car")
public class CarDoc {@Idprivate Long id;@Field(type = FieldType.Text, analyzer = "ik_smart", searchAnalyzer = "ik_smart")private String title;@Field(type = FieldType.Keyword)private String cover;@Field(type = FieldType.Double)private BigDecimal salePrice;@Field(type = FieldType.Double)private BigDecimal costPrice;@Field(type = FieldType.Integer)private Integer isNew;@Field(type = FieldType.Date)private Date registerTime;@Field(type = FieldType.Double)private Double mileAge;@Field(type = FieldType.Long)private Long shopId;@MultiField(mainField = @Field(type = FieldType.Text, analyzer = "ik_smart", searchAnalyzer = "ik_smart"),otherFields = {@InnerField(type = FieldType.Keyword, suffix = "keyword")})private String shopName; // @MultiField对同一字段应用不同的分析器或存储策略,这里既可以将该字段按照text进行拆分,也可以作为keyword进行查询@Field(type = FieldType.Keyword)private String shopAddress;@Field(type = FieldType.Date)private Date onSaleTime;@Field(type = FieldType.Integer)private Integer costEffective;@Field(type = FieldType.Integer)private Integer rushSale;@Field(type = FieldType.Integer)private Integer quasiNewCar;@Field(type = FieldType.Integer)private Integer transitiveCountry;@Field(type = FieldType.Long)private Long typeId;@MultiField(mainField = @Field(type = FieldType.Text, analyzer = "ik_smart", searchAnalyzer = "ik_smart"),otherFields = {@InnerField(type = FieldType.Keyword, suffix = "keyword")})private String typeName; @Field(type = FieldType.Text, analyzer = "ik_smart", searchAnalyzer = "ik_smart")private String carInfo;@GeoPointField   // 表示坐标类型private GeoPoint shopPoint;  
}

准备一个Repository接口

@Repository
public interface CarDocRepository extends ElasticsearchRepository<CarDoc, Long> {
}// 接口中的泛型,一个是文档对象的类型,一个是文档对象id的类型

使用spring-boot-starter-data-elasticsearch对es进行操作时也要按照es请求的格式进行操作

{"query": {"bool": {"must": {"match_all": {}},"filter": {"term": {"username": "Steven King"}}}}
}

查询

@Service
public class CarSearchServiceImpl implements ICarSearchService {@Autowiredprivate CarDocRepository repository;@Autowiredprivate HighlightResultMapper highlightResultMapper;@Autowiredprivate ElasticsearchTemplate elasticsearchTemplate;@Overridepublic PageList<CarDoc> search(CarSearchQuery query) {NativeSearchQueryBuilder builder = new NativeSearchQueryBuilder();BoolQueryBuilder boole = QueryBuilders.boolQuery(); // 外层的boolList<QueryBuilder> filter = boole.filter();		// 内层的fitlerList<QueryBuilder> must = boole.must();		// 内层的must// 查询"title", "typeName", "shopName", "carInfo"中有查询关键字的结果if (StringUtils.isNotBlank(query.getSearch())) {must.add(QueryBuilders.multiMatchQuery(query.getSearch(), "title", "typeName", "shopName", "carInfo"));}if (Objects.nonNull(query.getShopId())) {filter.add(QueryBuilders.termQuery("shopId", query.getShopId()));}// 进行范围查询if (Objects.nonNull(query.getMinPrice())) {filter.add(QueryBuilders.rangeQuery("costPrice").gte(query.getMinPrice()));}if (Objects.nonNull(query.getMaxPrice())) {filter.add(QueryBuilders.rangeQuery("costPrice").lte(query.getMaxPrice()));}// 按时间进行查找if (Objects.nonNull(query.getCarAge())) {Date date = DateUtils.addYears(new Date(), (-query.getCarAge()));if (query.getCarAgeType() == 1) {filter.add(QueryBuilders.rangeQuery("registerTime").lt(date.getTime()));}if (query.getCarAgeType() == 0) {filter.add(QueryBuilders.rangeQuery("registerTime").gte(date.getTime()));}}// 经纬度Double lon = query.getLongitude();Double lat = query.getLatitude();// 按照经纬度计算距离并按照距离进行查询if (Objects.nonNull(lon) && Objects.nonNull(lat) && Objects.nonNull(query.getDistance())) {GeoDistanceQueryBuilder point = QueryBuilders.geoDistanceQuery("shopPoint").point(lat, lon).distance(query.getDistance(), DistanceUnit.KILOMETERS);filter.add(point);}// 将bool加入到最外层query的构造器builder.withQuery(boole);SortOrder order = SortOrder.ASC;if (!"asc".equals(query.getSortType())) {order = SortOrder.DESC;}// 通过指定字段进行排序if (Objects.nonNull(query.getSortField())) {FieldSortBuilder sort = SortBuilders.fieldSort(query.getSortField()).order(order);builder.withSort(sort);} else {// 没传则默认按照距离排序if (Objects.nonNull(lat) && Objects.nonNull(lon)) {GeoDistanceSortBuilder point = new GeoDistanceSortBuilder("shopPoint", lat, lon);GeoDistanceSortBuilder sort = point.order(order);builder.withSort(sort);}}// 高亮展示,通过给字段前后加html标签的方式实现高亮等效果,需要一个高亮的工具类HighlightBuilder.Field title = new HighlightBuilder.Field("title").preTags("<span style='color:red'>").postTags("</span>");builder.withHighlightFields(title);// 进行分页展示builder.withPageable(PageRequest.of(query.getCurrentPage() - 1, query.getPageSize()));TermsAggregationBuilder aggBuilders1 = AggregationBuilders.terms("typeIdGroup").field("typeId").order(BucketOrder.count(true)).subAggregation(AggregationBuilders.terms("typeNameGroup").field("typeName.keyword").order(BucketOrder.count(true)));// 聚合查询builder.addAggregation(aggBuilders1);AggregatedPage<CarDoc> carDocs = elasticsearchTemplate.queryForPage(builder.build(), CarDoc.class, highlightResultMapper);Map<String, Object> map = SearchUtil.handleTermsAggsData(carDocs.getAggregations());Long counts = carDocs.getTotalElements();List<CarDoc> content = carDocs.getContent();for (CarDoc doc : content) {String s = doc.getShopAddress();if (StringUtils.isNotBlank(s)) {String string = s.split("市")[0] + "市";}doc.setShopAddress(s);}return new PageList<>(counts, content, map);}
}

PageList

@Data
public class PageList<T> {private Long count;private List<T> data; // 用于存放聚合查询的数据private Map<String,Object> map;public PageList() {}public PageList(Long count, List<T> data, Map<String, Object> map) {this.count = count;this.data = data;this.map = map;}public PageList(Long count, List<T> data) {this.count = count;this.data = data;}
}

高亮结果映射器

@Component
public class HighlightResultMapper implements SearchResultMapper {@Overridepublic <T> AggregatedPage<T> mapResults(SearchResponse response, Class<T> aClass, Pageable pageable) {// 记录总条数long totalHits = response.getHits().getTotalHits();// 记录列表(泛型) - 构建Aggregate使用List<T> list = new ArrayList<>();// 获取搜索结果(真正的的记录)SearchHits hits = response.getHits();for (SearchHit hit : hits) {if(hits.getHits().length <= 0){return null;}// 将原本的JSON对象转换成Map对象Map<String, Object> map = hit.getSourceAsMap();// 获取高亮的字段MapMap<String, HighlightField> highlightFields = hit.getHighlightFields();for (Map.Entry<String, HighlightField> highlightField : highlightFields.entrySet()) {// 获取高亮的KeyString key = highlightField.getKey();// 获取高亮的ValueHighlightField value = highlightField.getValue();// 实际fragments[0]就是高亮的结果,无需遍历拼接Text[] fragments = value.getFragments();StringBuilder sb = new StringBuilder();for (Text text : fragments) {sb.append(text);}// 因为高亮的字段必然存在于Map中,就是key值// 可能有一种情况,就是高亮的字段是嵌套Map,也就是说在Map里面还有Map的这种情况,这里没有考虑map.put(key, sb.toString());}// 把Map转换成对象T item = JSONObject.parseObject(JSONObject.toJSONString(map),aClass);list.add(item);}// 返回的是带分页的结果return new AggregatedPageImpl<>(list, pageable, totalHits,response.getAggregations()); //获取聚合结果}@Overridepublic <T> T mapSearchHit(SearchHit searchHit, Class<T> aClass) {return null;}
}

搜索工具类,用于处理聚合查询结果

public class SearchUtil {/*** 处理terms聚合   id  name* @param aggregations* @return*/public static Map<String, Object> handleTermsAggsData(Aggregations aggregations) {// 获取聚合查询结果Map<String, Aggregation> aggregationsMap = aggregations.getAsMap();Set<Map.Entry<String, Aggregation>> entries = aggregationsMap.entrySet();Iterator<Map.Entry<String, Aggregation>> iterator = entries.iterator();//6.1有多少聚合就要返回多少个key-List<IdName>Map<String, Object> aggsData = new HashMap<>();while (iterator.hasNext()) {Map.Entry<String, Aggregation> entry = iterator.next();String key = entry.getKey();System.out.println(key);Aggregation aggsId = entry.getValue();if (aggsId instanceof LongTerms) {   //6.2 拿到id聚合,并且必须是LongTermsLongTerms aggsIdLong = (LongTerms) aggsId;List<LongTerms.Bucket> buckets = aggsIdLong.getBuckets();//6.3 List<IdName<Long,String>List<IdName> list = new ArrayList<>();buckets.forEach(bucket -> {String idStr = bucket.getKeyAsString();//6.4 通过子聚合获取nameMap<String, Aggregation> subAggs = bucket.getAggregations().getAsMap();Set<Map.Entry<String, Aggregation>> entries1 = subAggs.entrySet();//直接获取第一个Map.Entry<String, Aggregation> nameAggEntry = entries1.iterator().next();Aggregation nameAgg = nameAggEntry.getValue();if (nameAgg instanceof StringTerms) {StringTerms nameAggStringTerms = (StringTerms) nameAgg;String nameStr = nameAggStringTerms.getBuckets().get(0).getKeyAsString();IdName idName = new IdName();idName.setId(Long.valueOf(idStr));idName.setName(nameStr);list.add(idName);}});aggsData.put(key, list);}}return aggsData;}
}

相关文章:

SpringBoot集成Elasticsearch实例

SpringBoot项目集成Elasticsearch实例 导包 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-elasticsearch</artifactId> </dependency>配置es连接 spring:data:elasticsearch:cluster…...

2024.ZCPC.M题 计算三角形个数

题目描述&#xff1a; 小蔡有一张三角形的格子纸&#xff0c;上面有一个大三角形。这个边长为 的大三角形&#xff0c; 被分成 个边长为 1 的小三角形(如图一所示)。现在&#xff0c;小蔡选择了一条水平边 删除&#xff08;如图二所示&#xff09;&#xff0c;请你找出图上剩余…...

Java常见设计模式入门与实践

设计模式是软件开发中被反复应用的、为解决特定问题而总结出的最佳实践。它们提供了开发可重用、灵活和高效软件系统的方法。在Java中&#xff0c;设计模式可以帮助开发者编写更高质量的代码。以下是Java中一些常用设计模式的入门介绍及其实践示例。 1. 单例模式 (Singleton P…...

110.平衡二叉树

给定一个二叉树&#xff0c;判断它是否是 平衡二叉树 题解&#xff1a;平衡二叉树 是指该树所有节点的左右子树的深度相差不超过 1。可以采用递归遍历每一个节点&#xff0c;得到其高度&#xff0c;在判断高度时不可避免的要用到其左右子树的高度&#xff0c;所以可以顺便判断出…...

字符串数组——传递文本的不同方法实例

一、实例化一个文本并在文本上直接显示接收的信息 1、制作一个预制体 (1) UI-Text (TMP)&#xff0c;TitleText (2) 给Text (TMP)添加Recipients.cs组件 using System.Collections; using System.Collections.Generic; using TMPro; using UnityEngine;public class Recipi…...

JDBC简介

JDBC最基本的知识点要求理解并会使用下边的两个案例即可&#xff1a; 前言&#xff1a; Jdbc操作步骤简析&#xff1a; 1.Connection connDriverManager.getConnection(url,账号&#xff0c;密码)&#xff1a;用DriverManager类的连接方法根据url账号密码连接数据库&#xf…...

RK3568平台(触摸篇)触摸屏基本原理

一.触摸屏概述 触摸屏作为一种新的输入设备&#xff0c;它是目前最简单、方便、自然的一种人机交互方式。 触摸屏又称为“触控屏”、“触控面板”&#xff0c;是一种可接收触头等输入讯号的感应式液晶显示装置&#xff1b;当接触了屏幕上的图形按钮时&#xff0c;屏幕上的触觉…...

【太原理工大学】软件系统安全—分析题

OK了&#xff0c;又是毫无准备的一场仗&#xff0c;我真是ありがとうございます 凸^o^凸 根据前几年传下来的信息&#xff0c;所谓“分析”&#xff0c;就是让你根据情节自行设计&#xff0c;例如如何设计表单等&#xff0c;这类多从实验中出&#xff0c;王老师强调好好做实验一…...

【仪器仪表/电源专题】浮地信号的测试的四种方案对比

接地信号和浮地信号区别 所有的电压测量都是差分测量&#xff0c;差分测量定义为两点之间的电压差。所以会分成两类&#xff1a; 1.参考地电平测量&#xff08;有时也叫接地信号&#xff09; 2.非参考地电平测量&#xff08;也称为浮地测量&#xff09; 测试信号可以分为接地信…...

Centos7安装jdk8或11以及切换方案

目录 jdk安装 安装OpenJDK11 安装OpenJDK8 配置默认的 Java 版本 验证 全局环境变量&#xff08;选配&#xff09; 个人版&#xff08;自己可以用&#xff0c;公司不建议&#xff09; 公司版本 /etc/profile 和 ~/.bash_profile 区别 前言-与正文无关 生活远不止眼前的苦…...

计算机二级Access选择题考点—代码篇

第1题 窗体的事件过程如下: Private Sub Form MouseDown(Button As Integer ,Shift As integer,X As Single,Y As Single)If Shift 6 And Button 2 Then MsgBox "Hello World."End lf End Sub程序运行后&#xff0c;要在窗体消息框中显示"Hello World."…...

海外仓系统如何让海外仓受益,WMS海外仓系统使用指南

随着跨境电商业务的快速发展&#xff0c;海外仓面临着需要更加高速运转的巨大挑战。 当海外仓出现因为手动作业导致效率低下&#xff0c;成本不断飙升或者出现库存管理问题的时候&#xff0c;意味着是时候引入一套合适的海外仓管理系统了。 对于寻求海外仓业务流程优化的企业…...

贪心-区间问题

135. 分发糖果 问题描述 n 个孩子站成一排。给你一个整数数组 ratings 表示每个孩子的评分。 你需要按照以下要求&#xff0c;给这些孩子分发糖果&#xff1a; 每个孩子至少分配到 1 个糖果。相邻两个孩子评分更高的孩子会获得更多的糖果。 请你给每个孩子分发糖果&#x…...

算法分析与设计期末考试复习GDPU

重点内容&#xff1a; 绪论&#xff1a; 简单的递推方程求解 1.19(1)(2) 、 教材例题 多个函数按照阶的大小排序 1.18 分治法&#xff1a; 分治法解决芯片测试问题 计算a^n的复杂度为logn的算法&#xff08;快速幂&#xff09; 分治法解决平面最近点对问…...

分批次训练和评估神经网络模型

【背景】 训练神经网络模型的时候&#xff0c;特征组合太多&#xff0c;电脑的资源会不足&#xff0c;所以采用分批逐步进行。已经处理过的批次保存下来&#xff0c;在下一次跳过&#xff0c;只做新加入的批次训练。 选择最优模型组合在中间结果的范围内选择&#xff0c;这样…...

【CS.AL】算法核心之分治算法:从入门到进阶

文章目录 1. 概述2. 适用场景3. 设计步骤4. 优缺点5. 典型应用6. 题目和代码示例6.1 简单题目&#xff1a;归并排序6.2 中等题目&#xff1a;最近点对问题6.3 困难题目&#xff1a;分数背包问题 7. 题目和思路表格8. 总结References 1000.01.CS.AL.1.4-核心-DivedeToConquerAlg…...

leetcode刷题记录:hot100强化训练2:二叉树+图论

二叉树 36. 二叉树的中序遍历 递归就不写了&#xff0c;写一下迭代法 class Solution(object):def inorderTraversal(self, root):""":type root: TreeNode:rtype: List[int]"""if not root:return res []cur rootstack []while cur or st…...

湘潭大学信息与网络安全复习笔记2(总览)

前面的实验和作业反正已经结束了&#xff0c;现在就是集中火力把剩下的内容复习一遍&#xff0c;这一篇博客的内容主要是参考教学大纲和教学日历 文章目录 教学日历教学大纲 教学日历 总共 12 次课&#xff0c;第一次课是概述&#xff0c;第二次和第三次课是密码学基础&#x…...

C语言:头歌使用函数找出数组中的最大值

任务描述 本关任务&#xff1a;本题要求实现一个找出整型数组中最大值的函数。 函数接口定义&#xff1a; int FindArrayMax( int a[], int n ); 其中a是用户传入的数组&#xff0c;n是数组a中元素的个数。函数返回数组a中的最大值。 主程序样例: #include <stdio.h>#…...

【技巧】Leetcode 191. 位1的个数【简单】

位1的个数 编写一个函数&#xff0c;输入是一个无符号整数&#xff08;以二进制串的形式&#xff09;&#xff0c;返回其二进制表达式中 设置位 的个数&#xff08;也被称为汉明重量&#xff09;。 示例 1&#xff1a; 输入&#xff1a;n 11 输出&#xff1a;3 解释&#x…...

C++_核心编程_多态案例二-制作饮品

#include <iostream> #include <string> using namespace std;/*制作饮品的大致流程为&#xff1a;煮水 - 冲泡 - 倒入杯中 - 加入辅料 利用多态技术实现本案例&#xff0c;提供抽象制作饮品基类&#xff0c;提供子类制作咖啡和茶叶*//*基类*/ class AbstractDr…...

(十)学生端搭建

本次旨在将之前的已完成的部分功能进行拼装到学生端&#xff0c;同时完善学生端的构建。本次工作主要包括&#xff1a; 1.学生端整体界面布局 2.模拟考场与部分个人画像流程的串联 3.整体学生端逻辑 一、学生端 在主界面可以选择自己的用户角色 选择学生则进入学生登录界面…...

CMake基础:构建流程详解

目录 1.CMake构建过程的基本流程 2.CMake构建的具体步骤 2.1.创建构建目录 2.2.使用 CMake 生成构建文件 2.3.编译和构建 2.4.清理构建文件 2.5.重新配置和构建 3.跨平台构建示例 4.工具链与交叉编译 5.CMake构建后的项目结构解析 5.1.CMake构建后的目录结构 5.2.构…...

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...

深入理解JavaScript设计模式之单例模式

目录 什么是单例模式为什么需要单例模式常见应用场景包括 单例模式实现透明单例模式实现不透明单例模式用代理实现单例模式javaScript中的单例模式使用命名空间使用闭包封装私有变量 惰性单例通用的惰性单例 结语 什么是单例模式 单例模式&#xff08;Singleton Pattern&#…...

Matlab | matlab常用命令总结

常用命令 一、 基础操作与环境二、 矩阵与数组操作(核心)三、 绘图与可视化四、 编程与控制流五、 符号计算 (Symbolic Math Toolbox)六、 文件与数据 I/O七、 常用函数类别重要提示这是一份 MATLAB 常用命令和功能的总结,涵盖了基础操作、矩阵运算、绘图、编程和文件处理等…...

自然语言处理——循环神经网络

自然语言处理——循环神经网络 循环神经网络应用到基于机器学习的自然语言处理任务序列到类别同步的序列到序列模式异步的序列到序列模式 参数学习和长程依赖问题基于门控的循环神经网络门控循环单元&#xff08;GRU&#xff09;长短期记忆神经网络&#xff08;LSTM&#xff09…...

mysql已经安装,但是通过rpm -q 没有找mysql相关的已安装包

文章目录 现象&#xff1a;mysql已经安装&#xff0c;但是通过rpm -q 没有找mysql相关的已安装包遇到 rpm 命令找不到已经安装的 MySQL 包时&#xff0c;可能是因为以下几个原因&#xff1a;1.MySQL 不是通过 RPM 包安装的2.RPM 数据库损坏3.使用了不同的包名或路径4.使用其他包…...

招商蛇口 | 执笔CID,启幕低密生活新境

作为中国城市生长的力量&#xff0c;招商蛇口以“美好生活承载者”为使命&#xff0c;深耕全球111座城市&#xff0c;以央企担当匠造时代理想人居。从深圳湾的开拓基因到西安高新CID的战略落子&#xff0c;招商蛇口始终与城市发展同频共振&#xff0c;以建筑诠释对土地与生活的…...

【Nginx】使用 Nginx+Lua 实现基于 IP 的访问频率限制

使用 NginxLua 实现基于 IP 的访问频率限制 在高并发场景下&#xff0c;限制某个 IP 的访问频率是非常重要的&#xff0c;可以有效防止恶意攻击或错误配置导致的服务宕机。以下是一个详细的实现方案&#xff0c;使用 Nginx 和 Lua 脚本结合 Redis 来实现基于 IP 的访问频率限制…...