尚品汇-ES(三十一)
目录:
(1)封装搜索相关实体对象
(2)搜索接口封装
(3)在service-list-client模块添加远程接口
(1)封装搜索相关实体对象
搜索参数实体:SearchParam
搜索参数实体:SearchParam
package com.atguigu.gmall.model.list;
/*** 商品搜索参数* 参数说明:* 1,商标品牌:trademark=2:华为 * 2:为品牌id,搜索字段* 华为:品牌名称,页面回显属性* 2,平台属性:props=23:4G:运行内存* 23:平台属性id,搜索字段* 运行内存:平台属性名称,页面回显属性* 4G:平台属性值,搜索字段与页面回显属性* </p>**/
@Data
public class SearchParam {// ?category3Id=61&trademark=2:华为&props=23:4G:运行内存&order=1:desc//category3Id=61private Long category1Id;;//三级分类idprivate Long category2Id;private Long category3Id;//trademark=2:华为private String trademark;//品牌idprivate String keyword;//检索的关键字// order=1:asc 排序规则 0:ascprivate String order = "";// 1:综合排序/热点 2:价格//props=23:4G:运行内存private String[] props;//页面提交的数组private Integer pageNo = 1;//分页信息private Integer pageSize = 12;
}
搜索结果集实体:SearchResponseVo
搜索结果集实体:SearchResponseVopackage com.atguigu.gmall.model.list;@Data
public class SearchResponseVo implements Serializable {//品牌 此时vo对象中的id字段保留(不用写) name就是“品牌” value: [{id:100,name:华为,logo:xxx},{id:101,name:小米,log:yyy}]private List<SearchResponseTmVo> trademarkList;//所有商品的顶头显示的筛选属性private List<SearchResponseAttrVo> attrsList = new ArrayList<>();//检索出来的商品信息private List<Goods> goodsList = new ArrayList<>();private Long total;//总记录数private Integer pageSize;//每页显示的内容private Integer pageNo;//当前页面private Long totalPages;}
结果集品牌实体:SearchResponseTmVo
package com.atguigu.gmall.model.list;@Data
public class SearchResponseTmVo implements Serializable {//当前属性值的所有值private Long tmId;//属性名称private String tmName;//网络制式,分类//图片urlprivate String tmLogoUrl;
}
结果集平台属性实体:SearchResponseAttrVo
package com.atguigu.gmall.model.list;@Data
public class SearchResponseAttrVo implements Serializable {private Long attrId;//1//当前属性值的所有值private List<String> attrValueList = new ArrayList<>();//属性名称private String attrName;//网络制式,分类
}
(2)搜索接口封装
SearchService接口
/*** 搜索列表* @param searchParam* @return* @throws IOException*/
SearchResponseVo search(SearchParam searchParam) throws IOException;
接口实现类
api参考文档:
Java REST Client [7.8] | Elastic
Java REST Client [7.8] | Elastic
@Autowired
private RestHighLevelClient restHighLevelClient;
@Override
public SearchResponseVo search(SearchParam searchParam) throws IOException {// 构建dsl语句SearchRequest searchRequest = this.buildQueryDsl(searchParam);SearchResponse response = this.restHighLevelClient.search(searchRequest, RequestOptions.DEFAULT);System.out.println(response);SearchResponseVo responseVO = this.parseSearchResult(response);responseVO.setPageSize(searchParam.getPageSize());responseVO.setPageNo(searchParam.getPageNo());long totalPages = (responseVO.getTotal()+searchParam.getPageSize()-1)/searchParam.getPageSize();responseVO.setTotalPages(totalPages);return responseVO;
}//封装查询条件
// 制作dsl 语句
private SearchRequest buildQueryDsl(SearchParam searchParam) {// 构建查询器SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder();// 构建多条件对象boolQueryBuilderBoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();// 判断查询条件是否为空 关键字if (!StringUtils.isEmpty(searchParam.getKeyword())){// 小米手机 小米and手机// MatchQueryBuilder matchQueryBuilder = new MatchQueryBuilder("title",searchParam.getKeyword()).operator(Operator.AND);MatchQueryBuilder title = QueryBuilders.matchQuery("title", searchParam.getKeyword()).operator(Operator.AND);boolQueryBuilder.must(title);}// 构建品牌查询String trademark = searchParam.getTrademark();if (!StringUtils.isEmpty(trademark)){// trademark=2:华为String[] split = StringUtils.split(trademark, ":");if (split != null && split.length == 2) {//构建过滤品牌TermQueryBuilder tmId=QueryBuilders.termQuery("tmId", split[0]);// 根据品牌Id过滤 添加到多条件对象boolQueryBuilder.filter(tmId);}}// 构建分类过滤 用户在点击的时候,只能点击一个值,所以此处使用termif(null!=searchParam.getCategory1Id()){boolQueryBuilder.filter(QueryBuilders.termQuery("category1Id",searchParam.getCategory1Id()));}// 构建分类过滤if(null!=searchParam.getCategory2Id()){boolQueryBuilder.filter(QueryBuilders.termQuery("category2Id",searchParam.getCategory2Id()));}// 构建分类过滤if(null!=searchParam.getCategory3Id()){boolQueryBuilder.filter(QueryBuilders.termQuery("category3Id",searchParam.getCategory3Id()));}// 构建平台属性查询// 23:4G:运行内存String[] props = searchParam.getProps();if (props!=null && props.length>0){// 循环遍历for (String prop : props) {// 23:4G:运行内存 平台属性id:平台属性值名称:平台属性名String[] split = StringUtils.split(prop, ":");if (split!=null && split.length==3){// 构建嵌套查询 创建多条件对象BoolQueryBuilder boolQuery = QueryBuilders.boolQuery();// 嵌套查询子查询BoolQueryBuilder subBoolQuery = QueryBuilders.boolQuery();// 构建子查==询中的过滤条件subBoolQuery.must(QueryBuilders.termQuery("attrs.attrId",split[0]));subBoolQuery.must(QueryBuilders.termQuery("attrs.attrValue",split[1]));// ScoreMode.None ?boolQuery.must(QueryBuilders.nestedQuery("attrs",subBoolQuery, ScoreMode.None));// 添加到整个过滤对象中,外层对象boolQueryBuilder.filter(boolQuery);}}}// 执行查询方法searchSourceBuilder.query(boolQueryBuilder);// 构建分页int from = (searchParam.getPageNo()-1)*searchParam.getPageSize();searchSourceBuilder.from(from);searchSourceBuilder.size(searchParam.getPageSize());// 排序 1:hotScore 2:price 1:综合排序/热度 2:价格//1:ascString order = searchParam.getOrder();if (!StringUtils.isEmpty(order)){// 判断排序规则String[] split = StringUtils.split(order, ":");if (split!=null && split.length==2){// 排序的字段String field = null;// 数组中的第一个参数switch (split[0]){case "1":field="hotScore";break;case "2":field="price";break;}searchSourceBuilder.sort(field,"asc".equals(split[1])? SortOrder.ASC:SortOrder.DESC);}else {// 没有传值的时候给默认值searchSourceBuilder.sort("hotScore",SortOrder.DESC);}}// 构建高亮HighlightBuilder highlightBuilder = new HighlightBuilder();highlightBuilder.field("title");highlightBuilder.postTags("</span>");highlightBuilder.preTags("<span style=color:red>");searchSourceBuilder.highlighter(highlightBuilder);// 设置品牌聚合TermsAggregationBuilder termsAggregationBuilder = AggregationBuilders.terms("tmIdAgg").field("tmId").subAggregation(AggregationBuilders.terms("tmNameAgg").field("tmName")).subAggregation(AggregationBuilders.terms("tmLogoUrlAgg").field("tmLogoUrl"));searchSourceBuilder.aggregation(termsAggregationBuilder);// 设置平台属性聚合searchSourceBuilder.aggregation(AggregationBuilders.nested("attrAgg", "attrs").subAggregation(AggregationBuilders.terms("attrIdAgg").field("attrs.attrId").subAggregation(AggregationBuilders.terms("attrNameAgg").field("attrs.attrName")).subAggregation(AggregationBuilders.terms("attrValueAgg").field("attrs.attrValue"))));// 结果集过滤searchSourceBuilder.fetchSource(new String[]{"id","defaultImg","title","price"},null);SearchRequest searchRequest = new SearchRequest("goods");//searchRequest.types("_doc");//将构建对象添加到请求中searchRequest.source(searchSourceBuilder);System.out.println("dsl:"+searchSourceBuilder.toString());return searchRequest;
}// 制作返回结果集
private SearchResponseVo parseSearchResult(SearchResponse response) {SearchHits hits = response.getHits();//声明对象SearchResponseVo searchResponseVo = new SearchResponseVo();//获取品牌的集合Map<String, Aggregation> aggregationMap = response.getAggregations().asMap();//ParsedLongTerms ?ParsedLongTerms tmIdAgg = (ParsedLongTerms) aggregationMap.get("tmIdAgg");List<SearchResponseTmVo> trademarkList = tmIdAgg.getBuckets().stream().map(bucket -> {SearchResponseTmVo trademark = new SearchResponseTmVo();//获取品牌Idtrademark.setTmId((Long.parseLong(((Terms.Bucket) bucket).getKeyAsString())));//trademark.setTmId(Long.parseLong(bucket.getKeyAsString()));//获取品牌名称Map<String, Aggregation> tmIdSubMap = ((Terms.Bucket) bucket).getAggregations().asMap();ParsedStringTerms tmNameAgg = (ParsedStringTerms) tmIdSubMap.get("tmNameAgg");String tmName = tmNameAgg.getBuckets().get(0).getKeyAsString();trademark.setTmName(tmName);
ParsedStringTerms tmLogoUrlAgg = (ParsedStringTerms) tmIdSubMap.get("tmLogoUrlAgg");
String tmLogoUrl = tmLogoUrlAgg.getBuckets().get(0).getKeyAsString();
trademark.setTmLogoUrl(tmLogoUrl);return trademark;}).collect(Collectors.toList());searchResponseVo.setTrademarkList(trademarkList);//赋值商品列表SearchHit[] subHits = hits.getHits();List<Goods> goodsList = new ArrayList<>();if (subHits!=null && subHits.length>0){//循环遍历for (SearchHit subHit : subHits) {// 将subHit 转换为对象Goods goods = JSONObject.parseObject(subHit.getSourceAsString(), Goods.class);//获取高亮if (subHit.getHighlightFields().get("title")!=null){Text title = subHit.getHighlightFields().get("title").getFragments()[0];goods.setTitle(title.toString());}goodsList.add(goods);}}searchResponseVo.setGoodsList(goodsList);//获取平台属性数据ParsedNested attrAgg = (ParsedNested) aggregationMap.get("attrAgg");ParsedLongTerms attrIdAgg = attrAgg.getAggregations().get("attrIdAgg");List<? extends Terms.Bucket> buckets = attrIdAgg.getBuckets();if (!CollectionUtils.isEmpty(buckets)){List<SearchResponseAttrVo> searchResponseAttrVOS = buckets.stream().map(bucket -> {//声明平台属性对象SearchResponseAttrVo responseAttrVO = new SearchResponseAttrVo();//设置平台属性值IdresponseAttrVO.setAttrId(((Terms.Bucket) bucket).getKeyAsNumber().longValue());ParsedStringTerms attrNameAgg = bucket.getAggregations().get("attrNameAgg");List<? extends Terms.Bucket> nameBuckets = attrNameAgg.getBuckets();responseAttrVO.setAttrName(nameBuckets.get(0).getKeyAsString());//设置规格参数列表ParsedStringTerms attrValueAgg = ((Terms.Bucket) bucket).getAggregations().get("attrValueAgg");List<? extends Terms.Bucket> valueBuckets = attrValueAgg.getBuckets();List<String> values = valueBuckets.stream().map(Terms.Bucket::getKeyAsString).collect(Collectors.toList());responseAttrVO.setAttrValueList(values);return responseAttrVO;}).collect(Collectors.toList());searchResponseVo.setAttrsList(searchResponseAttrVOS);}// 获取总记录数searchResponseVo.setTotal(hits.getTotalHits().value);return searchResponseVo;
}
控制器ListApiController
/*** 搜索商品* @param searchParam* @return* @throws IOException*/
@PostMapping
public Result list(@RequestBody SearchParam searchParam) throws IOException {SearchResponseVo response = searchService.search(searchParam);return Result.ok(response);
}
在service-list 模块中配置logstash
首先在service模块中添加依赖
<dependency>
<groupId>net.logstash.logback</groupId>
<artifactId>logstash-logback-encoder</artifactId>
<version>5.1</version>
</dependency>
其次,将日志配置文件放入到resources目录下!
(3)在service-list-client模块添加远程接口
package com.atguigu.gmall.list.client;
@FeignClient(value = "service-list", fallback = ListDegradeFeignClient.class)
public interface ListFeignClient {/*** 搜索商品* @param listParam* @return*/@PostMapping("/api/list")Result list(@RequestBody SearchParam listParam);/*** 上架商品* @param skuId* @return*/@GetMapping("/api/list/inner/upperGoods/{skuId}")Result upperGoods(@PathVariable("skuId") Long skuId);/*** 下架商品* @param skuId* @return*/@GetMapping("/api/list/inner/lowerGoods/{skuId}")Result lowerGoods(@PathVariable("skuId") Long skuId);}
package com.atguigu.gmall.list.client.impl;@Component
public class ListDegradeFeignClient implements ListFeignClient {@Overridepublic Result list(SearchParam searchParam) {return Result.fail();}@Overridepublic Result upperGoods(Long skuId) {return null;}@Overridepublic Result lowerGoods(Long skuId) {return null;}
}
相关文章:

尚品汇-ES(三十一)
目录: (1)封装搜索相关实体对象 (2)搜索接口封装 (3)在service-list-client模块添加远程接口 (1)封装搜索相关实体对象 搜索参数实体:SearchParam 搜索参…...

NC 跳台阶
系列文章目录 文章目录 系列文章目录前言 前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站,这篇文章男女通用,看懂了就去分享给你的码吧。 描述 一只青蛙一次…...

linux 文件编程
1. 标准IO 也称为标准输入输出(Standard Input/Output),是计算机编程中一种常见的IO操作方式,特别是在C语言及其衍生语言中广泛使用。它主要通过标准C库中的函数来实现,提供了丰富的接口用于数据的输入和输出。 文本文…...

【后端速成 Vue】实现动态表白墙
前言: 通过前面几篇的文章的讲解,已经学习到了很多的 Vue 指令了,那么现在就将学习到的指令利用起来,做一个小的 demo。 最终效果图: 通过效果图可以发现,一共有这几个功能: ● 渲染列表&…...

【日常开发】 java返回ECharts数据结构封装
java返回ECharts数据结构封装 一、前端页面示例图如下: 二、准备测试数据: 三、后端 格式封装代码: 四、最终结果: 🎈边走、边悟🎈迟早会好 一、前端页面示例图如下: 二、准备测试数据&am…...

Table表格控件实现单选功能
Table表格控件实现单选功能 <el-tableref"tableRef"height"385"style"--el-table-border-color: none"row-key"contractId"highlight-current-rowsingle-selectselect"handleSelect":data"contractInfo">&l…...

AI技术加速落地 港科广联手思谋打开智能缺陷检测新纪元
AI 技术应用落地的元年,工业是主战场,尤其是工业缺陷检测。 在“生产制造-缺陷检测-工艺优化-生产制造”的智能制造闭环链条中,基于AI的智能缺陷检测扮演着“把关者”的角色。但这个把关者长期以来却缺少一个称手的工具——样本量大、精度高…...

Python爬虫开发:BeautifulSoup、Scrapy入门
在现代网络开发中,网络爬虫是一个非常重要的工具。它可以自动化地从网页中提取数据,并且可以用于各种用途,如数据收集、信息聚合和内容监控等。在Python中,有多个库可以用于爬虫开发,其中BeautifulSoup和Scrapy是两个非…...

数据科学、数据分析、人工智能必备知识汇总-----常用数据分析方法-----持续更新
数据科学、数据分析、人工智能必备知识汇总-----主目录-----持续更新(进不去说明我没写完):https://blog.csdn.net/grd_java/article/details/140174015 文章目录 一、对比分析法1. 按时间和地区2. 同比和环比 二、分组分析法三、结构分析法四、交叉分析法五、矩阵分…...
学习vue Router 一 起步,编程式导航,历史记录,路由传参
目录 起步,安装 1. 安装 2. 使用 命名路由 编程式导航 1. 字符串模式 2. 对象模式 3. 命名路由模式 历史记录 replace的使用 横跨历史 路由传参 1. query路由传参 2. 动态路由传参 3. 二者的区别 起步,安装 router 路由 因为vue是单页应用…...

Qt/C++最新地图组件发布/历时半年重构/同时支持各种地图内核/包括百度高德腾讯天地图
一、前言说明 最近花了半年时间,专门重构了整个地图组件,之前写的比较粗糙,有点为了完成功能而做的,没有考虑太多拓展性和易用性。这套地图自检这几年大量的实际项目和用户使用下来,反馈了不少很好的建议和意见&#…...
Laravel + Thinkphp 生成二维码
安装依赖 composer require endroid/qr-code 编写ThinkPhP代码 public function index() {// 创建二维码内容$qrCode new QrCode(Hello World);// 设置二维码的配置$qrCode->setSize(300);$qrCode->setMargin(10);// 获取二维码图像$writer new PngWriter();$result…...
2408C++,C++20的无侵入式反射
原文 C17基于结构绑定的编译期反射 事实上不需要宏的编译期反射在C17中已用得很多了,比如struct_pack的编译期反射就不需要宏,因为C17结构绑定可直接得到一个聚集类的成员的引用. struct person {int id;std::string name;int age; }; int main() {person p{1, "tom&qu…...
抽象工厂模式(Abstract factory pattern)- python实现
抽象工厂模式的通俗示例 想象一下,你正在经营一家家具店,你需要从不同的供应商那里采购不同的家具系列。有的供应商提供的是现代风格家具,包括现代沙发、现代椅子和现代桌子;而有的供应商提供的是古典风格家具,包括古…...
adb Connection reset by peer的解决方法
本文同步发于:https://www.cnblogs.com/yeshen-org/p/18350232 最近在编译一个老项目,项目中依赖了很多第三方库,用gradle编译要20-30分钟,而且内存开销很大。 公司配的15G内存的电脑,一次编译能用到14G。 编译的时候&…...
111111111
1111111111111111111...
搜维尔科技:Varjo XR-4使用UE5 打造最具沉浸感的混合现实环境
Varjo XR-4使用UE5打造最具沉浸感的混合现实环境 搜维尔科技:Varjo XR-4使用UE5 打造最具沉浸感的混合现实环境...

从分散到集中:TSINGSEE青犀EasyCVR视频汇聚网关在视频整体监控解决方案中的整合作用
边缘计算视频汇聚网关是基于开放式、大融合、全兼容、标准化的设计架构理念,依据《安全防范视频监控联网系统信息传输、交换、控制技术要求》(GB/T28181-2011)标准开发,集流媒体转发、视频编码、视频管理、标准通信协议、网络穿透…...

React学习-jsx语法
jsx语法,浏览器不认识,需要经过babel编译 https://babeljs.io/ 面试题:jsx的作用? 普通回答:可以在js中返回dom,经过babel编译成js认识的代码import { jsx as _jsx, jsxs as _jsxs } from "react/j…...

uniapp多图上传uni.chooseImage上传照片uni.uploadFile
uniapp多图上传uni.chooseImage上传照片uni.uploadFile 代码示例: /**上传照片 多图*/getImage() {uni.chooseImage({count: 9, //默认9sizeType: [original, compressed], //可以指定是原图还是压缩图,默认二者都有sourceType: [album], //从相册选择/…...
KubeSphere 容器平台高可用:环境搭建与可视化操作指南
Linux_k8s篇 欢迎来到Linux的世界,看笔记好好学多敲多打,每个人都是大神! 题目:KubeSphere 容器平台高可用:环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...
linux 错误码总结
1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...
【论文笔记】若干矿井粉尘检测算法概述
总的来说,传统机器学习、传统机器学习与深度学习的结合、LSTM等算法所需要的数据集来源于矿井传感器测量的粉尘浓度,通过建立回归模型来预测未来矿井的粉尘浓度。传统机器学习算法性能易受数据中极端值的影响。YOLO等计算机视觉算法所需要的数据集来源于…...

Cloudflare 从 Nginx 到 Pingora:性能、效率与安全的全面升级
在互联网的快速发展中,高性能、高效率和高安全性的网络服务成为了各大互联网基础设施提供商的核心追求。Cloudflare 作为全球领先的互联网安全和基础设施公司,近期做出了一个重大技术决策:弃用长期使用的 Nginx,转而采用其内部开发…...

Python爬虫(一):爬虫伪装
一、网站防爬机制概述 在当今互联网环境中,具有一定规模或盈利性质的网站几乎都实施了各种防爬措施。这些措施主要分为两大类: 身份验证机制:直接将未经授权的爬虫阻挡在外反爬技术体系:通过各种技术手段增加爬虫获取数据的难度…...

12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...
Unit 1 深度强化学习简介
Deep RL Course ——Unit 1 Introduction 从理论和实践层面深入学习深度强化学习。学会使用知名的深度强化学习库,例如 Stable Baselines3、RL Baselines3 Zoo、Sample Factory 和 CleanRL。在独特的环境中训练智能体,比如 SnowballFight、Huggy the Do…...

Unity | AmplifyShaderEditor插件基础(第七集:平面波动shader)
目录 一、👋🏻前言 二、😈sinx波动的基本原理 三、😈波动起来 1.sinx节点介绍 2.vertexPosition 3.集成Vector3 a.节点Append b.连起来 4.波动起来 a.波动的原理 b.时间节点 c.sinx的处理 四、🌊波动优化…...
稳定币的深度剖析与展望
一、引言 在当今数字化浪潮席卷全球的时代,加密货币作为一种新兴的金融现象,正以前所未有的速度改变着我们对传统货币和金融体系的认知。然而,加密货币市场的高度波动性却成为了其广泛应用和普及的一大障碍。在这样的背景下,稳定…...

SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...