商城-商品搜索(品牌统计、规格统计、条件筛选、搜索分页排序、高亮显示)

商城-商品搜索(品牌统计、规格统计、条件筛选、搜索分页排序、高亮显示)用户搜索的时候 除了使用分类搜索外 还有可能使用品牌搜索 所以我们还需要显示品牌数据和 规格数据 品牌数据和规格数据的显示比较容易 都可以考虑使用分类统计的方式进行分组实现 1 品牌统计 看下面的 SQL 语句 我们在执行搜索的时候 第 1 条 SQL 语句是执行搜

大家好,我是讯享网,很高兴认识大家。


讯享网

用户搜索的时候,除了使用分类搜索外,还有可能使用品牌搜索,所以我们还需要显示品牌数据和

规格数据,品牌数据和规格数据的显示比较容易,都可以考虑使用分类统计的方式进行分组实现。

1 品牌统计

看下面的SQL语句,我们在执行搜索的时候,第1条SQL语句是执行搜,第2条语句是根据品牌名字

分组查看有多少品牌,大概执行了2个步骤就可以获取数据结果以及品牌统计,我们可以发现他们

的搜索条件完全一样。

-- 查询所有 SELECT * FROM tb_sku WHERE name LIKE '%手机%'; -- 根据品牌名字分组查询 SELECT brand_name FROM tb_sku WHERE name LIKE '%手机%' GROUP BY brand_name;

讯享网

我们每次执行搜索的时候,需要显示商品品牌名称,这里要显示的品牌名称其实就是符合搜素条件

的所有商品的品牌集合,我们可以按照上面的实现思路,使用ES根据分组名称做一次分组查询即

可实现。

修改search微服务的com.changgou.search.service.impl.SkuServiceImpl类,

添加一个品牌分组搜索

添加的代码如下:

讯享网//设置分组条件 商品品牌 nativeSearchQueryBuilder.addAggregation(AggregationBuilders.terms("skuBrandgroup").field("brandName").size(50)); //获取分组结果 商品品牌 StringTerms stringTermsBrand = (StringTerms) skuPage.getAggregation("skuBrandgroup"); List<String> brandList = getStringsBrandList(stringTermsBrand); resultMap.put("brandList", brandList); / * 获取品牌列表 * * @param stringTermsBrand * @return */ private List<String> getStringsBrandList(StringTerms stringTermsBrand) { List<String> brandList = new ArrayList<>(); if (stringTermsBrand != null) { for (StringTerms.Bucket bucket : stringTermsBrand.getBuckets()) { brandList.add(bucket.getKeyAsString()); } } return brandList; }

测试:

http://localhost:18085/search

和下面规格一起测试

---------------------------------------------------------------------------------------------------------------------------------

2 规格统计

用户搜索的时候,除了使用分类、品牌搜索外,还有可能使用规格搜索,所以我们还需要显示规格

数据,规格数据的显示相比上面2种实现略微较难一些,需要对数据进行处理,我们也可以考虑使

用分类统计和品牌统计的方式进行分组实现。

看下面的SQL语句,我们在执行搜索的时候,第1条SQL语句是执行搜,第2条语句是根据规格分组

查看有多少规格,大概执行了2个步骤就可以获取数据结果以及规格统计,我们可以发现他们的搜

索条件完全一样。

-- 查询所有 SELECT * FROM tb_sku WHERE name LIKE '%手机%'; -- 根据规格名字分组查询 SELECT spec FROM tb_sku WHERE name LIKE '%手机%' GROUP BY spec;

上述SQL语句执行后的结果如下图:

获取到的规格数据我们发现有重复,不过也可以解决,解决思路如下:

讯享网1.获取所有规格数据 2.将所有规格数据转换成Map 3.定义一个Map<String,Set>,key是规格名字,防止重复所以用Map,valu是规格值,规格值有多个,所以用集合,为了防止规格重复,用Set去除重复 4.循环规格的Map,将数据填充到定义的Map<String,Set>中

我们每次执行搜索的时候,需要显示商品规格数据,这里要显示的规格数据其实就是符合搜素条件

的所有商品的规格集合,我们可以按照上面的实现思路,使用ES根据分组名称做一次分组查询,

并去除重复数据即可实现。

修改search微服务的com.changgou.search.service.impl.SkuServiceImpl类,添加一个规格分组搜

索:

//设置分组条件 商品的规格 nativeSearchQueryBuilder.addAggregation(AggregationBuilders.terms("skuSpecgroup").field("spec.keyword").size(100)); //获取分组结果 商品规格数据 StringTerms stringTermsSpec = (StringTerms) skuPage.getAggregation("skuSpecgroup"); Map<String, Set<String>> specMap = getStringSetMap(stringTermsSpec); resultMap.put("specMap", specMap); / * 获取规格列表数据 * * @param stringTermsSpec * @return */ private Map<String, Set<String>> getStringSetMap(StringTerms stringTermsSpec) { Map<String, Set<String>> specMap = new HashMap<String, Set<String>>(); Set<String> specList = new HashSet<>(); if (stringTermsSpec != null) { for (StringTerms.Bucket bucket : stringTermsSpec.getBuckets()) { specList.add(bucket.getKeyAsString()); } } for (String specjson : specList) { Map<String, String> map = JSON.parseObject(specjson, Map.class); for (Map.Entry<String, String> entry : map.entrySet()) {// String key = entry.getKey(); //规格名字 String value = entry.getValue(); //规格选项值 //获取当前规格名字对应的规格数据 Set<String> specValues = specMap.get(key); if (specValues == null) { specValues = new HashSet<String>(); } //将当前规格加入到集合中 specValues.add(value); //将数据存入到specMap中 specMap.put(key, specValues); } } return specMap; }

测试:

http://localhost:18085/search

。。。

---------------------------------------------------------------------------------------------------------------------------------

3 条件筛选

用户有可能会根据分类搜索、品牌搜索,还有可能根据规格搜索,以及价格搜索和排序操作。根据

分类和品牌搜索的时候,可以直接根据指定域搜索,而规格搜索的域数据是不确定的,价格是一个

区间搜索,所以我们可以分为三段时间,先实现分类、品牌搜素,再实现规格搜索,然后实现价格

区间搜索。

3.1 分类、品牌筛选

页面每次向后台传入对应的分类和品牌,后台据分类和品牌进行条件过滤即可。

修改搜索微服务com.changgou.search.service.impl.SkuServiceImpl的search方法,添加分类和品

牌过滤,添加过滤条件如下:

讯享网BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery(); if (!StringUtils.isEmpty(searchMap.get("brand"))) { boolQueryBuilder.filter(QueryBuilders.termQuery("brandName", searchMap.get("brand"))); } if (!StringUtils.isEmpty(searchMap.get("category"))) { boolQueryBuilder.filter(QueryBuilders.termQuery("categoryName", searchMap.get("category"))); } //构建过滤查询 nativeSearchQueryBuilder.withFilter(boolQueryBuilder); 

测试:

http://localhost:18085/search

。。。

---------------------------------------------------------------------------------------------------------------------------------

3.2 规格过滤

规格这一块,需要向后台发送规格名字以及规格值,我们可以按照一定要求来发送数据,例如规格

名字以特殊前缀提交到后台:spec_网络制式:电信4G、spec_显示屏尺寸:4.0-4.9英寸

后台接到数据后,可以根据前缀spec_来区分是否是规格,如果以spec_xxx开始的数据则为规格数

据,需要根据指定规格找信息。

上图是规格的索引存储格式,真实数据在spechMap.规格名字.keyword中,所以找数据也是按照如

下格式去找:

spechMap.规格名字.keyword

修改com.changgou.search.service.impl.SkuServiceImpl的search方法,增加规格查询操作,代码

如下:

//规格过滤查询 if (searchMap != null) { for (String key : searchMap.keySet()) { if (key.startsWith("spec_")) { boolQueryBuilder.filter(QueryBuilders.termQuery("specMap." + key.substring(5) + ".keyword", searchMap.get(key))); } } }

测试:

http://localhost:18085/search

。。。

---------------------------------------------------------------------------------------------------------------------------------

3.3 价格区间查询

价格区间查询,每次需要将价格传入到后台,前端传入后台的价格大概是price=0-500或者

price=500-1000依次类推,最后一个是price=3000,后台可以根据-分割,如果分割得到的结果最多

有2个,第1个表示x<price,第2个表示price<=y

讯享网//价格过滤查询 String price = searchMap.get("price"); if (!StringUtils.isEmpty(price)) { String[] split = price.split("-"); if (!split[1].equalsIgnoreCase("*")) { boolQueryBuilder.filter(QueryBuilders.rangeQuery("price").from(split[0], true).to(split[1], true)); } else { boolQueryBuilder.filter(QueryBuilders.rangeQuery("price").gte(split[0])); } }

测试:

http://localhost:18085/search

。。。

---------------------------------------------------------------------------------------------------------------------------------

4 搜索分页

页面需要实现分页搜索,所以我们后台每次查询的时候,需要实现分页。用户页面每次会传入当前

页和每页查询多少条数据,当然如果不传入每页显示多少条数据,默认查询30条即可。

分页使用PageRequest.of( pageNo- 1, pageSize);实现,第1个参数表示第N页,从0开始,第2个

参数表示每页显示多少条,实现代码如下:

//略 //构建过滤查询 nativeSearchQueryBuilder.withFilter(boolQueryBuilder); //构建分页查询 Integer pageNum = 1; if (!StringUtils.isEmpty(searchMap.get("pageNum"))) { try { pageNum = Integer.valueOf(searchMap.get("pageNum")); } catch (NumberFormatException e) { e.printStackTrace(); pageNum=1; } } Integer pageSize = 3; nativeSearchQueryBuilder.withPageable(PageRequest.of(pageNum - 1, pageSize)); //略 //4.构建查询对象 NativeSearchQuery query = nativeSearchQueryBuilder.build(); //略

测试:

http://localhost:18085/search

。。。

---------------------------------------------------------------------------------------------------------------------------------

5 搜索排序

排序这里总共有根据价格排序、根据评价排序、根据新品排序、根据销量排序,排序要想实现非常

简单,只需要告知排序的域以及排序方式即可实现。

价格排序:只需要根据价格高低排序即可,降序价格高->低,升序价格低->高

评价排序:评价分为好评、中评、差评,可以在数据库中设计3个列,用来记录好评、中评、差评

的量,每次排序的时候,好评的比例来排序,当然还要有条数限制,评价条数需要超过N条。

新品排序:直接根据商品的发布时间或者更新时间排序。

销量排序:销量排序除了销售数量外,还应该要有时间段限制。

这里我们不单独针对某个功能实现排序,我们只需要在后台接收2个参数,分别是排序域名字和排

序方式,代码如下:

解释: 前端页面传递要排序的字段(field)和要排序的类型(ASC,DESC),后台接收.

代码如下:

讯享网//构建排序查询 String sortRule = searchMap.get("sortRule"); String sortField = searchMap.get("sortField"); if (!StringUtils.isEmpty(sortRule) && !StringUtils.isEmpty(sortField)) { nativeSearchQueryBuilder.withSort(SortBuilders.fieldSort(sortField).order(sortRule.equals("DESC") ? SortOrder.DESC : SortOrder.ASC)); }

测试

根据价格降序:

{"keywords":"手机","pageNum":"1","sortRule":"DESC","sortField":"price"} 

根据价格升序:

讯享网{"keywords":"手机","pageNum":"1","sortRule":"ASC","sortField":"price"}

---------------------------------------------------------------------------------------------------------------------------------

6 高亮显示

高亮显示是指根据商品关键字搜索商品的时候,显示的页面对关键字给定了特殊样式,让它显示更

加突出,如上图商品搜索中,关键字编程了红色,其实就是给定了红色样式。

实现步骤:

将之前的搜索换掉,换成高亮搜索,我们需要做3个步骤:

1.指定高亮域,也就是设置哪个域需要高亮显示
  设置高亮域的时候,需要指定前缀和后缀,也就是关键词用什么html标签包裹,再给该标签样式
2.高亮搜索实现
3.将非高亮数据替换成高亮数据

第1点,例如在百度中搜索数据的时候,会有2个地方高亮显示,分别是标题和描述,商城搜索的时

候,只是商品名称高亮显示了。而高亮显示其实就是添加了样式,

例如<span style="color:red;">笔记本</span>,而其中span开始标签可以称为前缀,span结束标

签可以称为后缀。

第2点,高亮搜索使用ElasticsearchTemplate实现。

第3点,高亮搜索后,会搜出非高亮数据和高亮数据,高亮数据会加上第1点中的高亮样式,此时我

们需要将非高亮数据换成高亮数据即可。例如非高亮:华为笔记本性能超强悍 

高亮数据:华为<span style="color:red;"笔记本</span>性能超强悍,将非高亮的换成高亮的,到

页面就能显示样式了。

修改com.changgou.search.service.impl.SkuServiceImpl的search方法搜索代码,添加高亮显示的

域:

讯享网//设置高亮条件 nativeSearchQueryBuilder.withHighlightFields(new HighlightBuilder.Field("name")); nativeSearchQueryBuilder.withHighlightBuilder(new HighlightBuilder().preTags("<em style=\"color:red\">").postTags("</em>")); //设置主关键字查询 nativeSearchQueryBuilder.withQuery(QueryBuilders.multiMatchQuery(keywords,"name","brandName","categoryName"));

修改查询的方法,自定义结果映射器:

AggregatedPage<SkuInfo> skuPage = esTemplate.queryForPage(query, SkuInfo.class, new SearchResultMapperImpl());

自定义一个映射结果类实现接口,作用就是:自定义映射结果集,获取高亮的数据展示

讯享网public class SearchResultMapperImpl implements SearchResultMapper { @Override public <T> AggregatedPage<T> mapResults(SearchResponse response, Class<T> clazz, Pageable pageable) { List<T> content = new ArrayList<>(); //如果没有结果返回为空 if (response.getHits() == null || response.getHits().getTotalHits() <= 0) { return new AggregatedPageImpl<T>(content); } for (SearchHit searchHit : response.getHits()) { String sourceAsString = searchHit.getSourceAsString(); SkuInfo skuInfo = JSON.parseObject(sourceAsString, SkuInfo.class); Map<String, HighlightField> highlightFields = searchHit.getHighlightFields(); HighlightField highlightField = highlightFields.get("name"); //有高亮则设置高亮的值 if (highlightField != null) { StringBuffer stringBuffer = new StringBuffer(); for (Text text : highlightField.getFragments()) { stringBuffer.append(text.string()); } skuInfo.setName(stringBuffer.toString()); } content.add((T) skuInfo); } return new AggregatedPageImpl<T>(content, pageable, response.getHits().getTotalHits(), response.getAggregations(), response.getScrollId()); } }

测试:

http://localhost:18085/search

。。。

---------------------------------------------------------------------------------------------------------------------------------

小讯
上一篇 2025-02-17 17:00
下一篇 2025-03-28 09:16

相关推荐

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请联系我们,一经查实,本站将立刻删除。
如需转载请保留出处:https://51itzy.com/kjqy/33951.html