Elastic Search(ES)Java 入门实操(2)搜索代码

上篇解释了 ES 的基本概念和分词器。Elastic Search (ES)Java 入门实操(1)下载安装、概念-CSDN博客

Elastic Search(ES)Java 入门实操(3)数据同步-CSDN博客

这篇主要演示 Java 整合 ES进行数据搜索。

ES 实现搜索接口

首先根据 MySQL 字段,在 ES 创建索引。

create table mydb
(id          int auto_increment comment '序号'primary key,title       varchar(20)                        not null comment '标题',content     text                               not null comment '内容',cretateTime datetime default CURRENT_TIMESTAMP not null comment '创建时间',updateTime  datetime default CURRENT_TIMESTAMP not null on update CURRENT_TIMESTAMP comment '更新时间',isDelete    tinyint  default 0                 not null comment '是否删除'
)comment '文章' collate = utf8mb4_unicode_ci;
PUT article_1
{"aliases": {  #别名"article": {}},"mappings": {"properties": {"title": {"type": "text", #字段类型"analyzer": "ik_max_word",#插入时分词方式"search_analyzer": "ik_smart", #查询时分词方式"fields": { #字段配置,子字段"keyword": { "type": "keyword", #精确匹配"ignore_above": 256 #超过 256 字符就忽略查询}}},"content": {"type": "text","analyzer": "ik_max_word","search_analyzer": "ik_smart","fields": {"keyword": {"type": "keyword","ignore_above": 256}}},"createTime": {"type": "date"},"updateTime": {"type": "date"},"isDelete": {"type": "keyword"}}}
}

引入 spring-data-elasticsearch 依赖

需要注意版本号一定要兼容

Spring Data Elasticsearch - Reference Documentation

这里使用的是 7.17版本,所以选择相近的 4.4.x版本的 依赖,同样的 springboot 的版本也得严格对应。Maven Repository: org.springframework.data » spring-data-elasticsearch » 4.4.7 (mvnrepository.com)

 下面的报错就是版本不对,需要把springboot 改为 2.7+,

java.lang.NoSuchFieldError: INDEX_CONTENT_TYPE
<!-- https://mvnrepository.com/artifact/org.springframework.data/spring-data-elasticsearch -->
<dependency><groupId>org.springframework.data</groupId><artifactId>spring-data-elasticsearch</artifactId><version>4.4.7</version>
</dependency>

启动 springboot,有调用的日志

给 ES 索引创建实体类 

/*** ES 实体类* document 注解是将Java 对象映射到 Elasticsearch 索引和类型中*/
@Document(indexName = "article")
@Data
public class ArticleEsDto implements Serializable {private static final String DATE_TIME_PATTERN = "yyyy-MM-dd'T'HH:mm:ss.SSS'Z'";/*** id* 需要打上 id 注解,指定 ES 中存储的 id 是唯一字段* 如果新增是不传入,则 ES 会自动生成*/@Idprivate long id;/*** 标题*/private String title;/*** 内容*/private String content;/*** 创建时间*/@Field(index = false, store = true,type = FieldType.Date,format = {},pattern = DATE_TIME_PATTERN)private Date createTime;/*** 更新时间* Field:这是一个MyBatis-Plus注解,用于标注该字段在数据库表中的对应关系。* 其中,index = false表示该字段不在数据库表的索引中,* store = true表示该字段在数据库表的存储中,* type = FieldType.Date表示该字段的类型为Date,* format = {}表示该字段的格式为空,* pattern = DATE_TIME_PATTERN表示该字段的日期时间格式为DATE_TIME_PATTERN。*/@Field(index = false, store = true,type = FieldType.Date,format = {},pattern = DATE_TIME_PATTERN)private Date updateTime;/*** 是否删除*/private Integer isDelete;private static final long serialVersionUID = 1L;
}

第一种方式

elasticsearch Respository,新建类继承该类,默认提供了简单的增删改查方法,多用于可以预期的,相对不复杂的查询

@NoRepositoryBean
public interface CrudRepository<T, ID> extends Repository<T, ID> {<S extends T> S save(S entity);<S extends T> Iterable<S> saveAll(Iterable<S> entities);Optional<T> findById(ID id);boolean existsById(ID id);Iterable<T> findAll();Iterable<T> findAllById(Iterable<ID> ids);long count();void deleteById(ID id);void delete(T entity);void deleteAllById(Iterable<? extends ID> ids);void deleteAll(Iterable<? extends T> entities);void deleteAll();
}
/*** ES 的控制层* 继承 ElasticsearchRepository 即可*/
public interface ArticleEsDao extends ElasticsearchRepository<ArticleEsDto, Long> {// 这里可以扩展一些自定义方法// 例如:根据标题模糊查询List<ArticleEsDto> findByTitle(String title);}

新增测试

//注入接口@Resourceprivate ArticleEsDao articleEsDao;//测试新增@Testvoid EsTest1(){//创建实体对象并添加属性ArticleEsDto articleEsDto = new ArticleEsDto();articleEsDto.setId(1L);articleEsDto.setTitle("青花瓷");articleEsDto.setContent("天青色等烟雨而我在等你");articleEsDto.setCreateTime(new Date());articleEsDto.setUpdateTime(new Date());articleEsDto.setIsDelete(0);//调用方法保存articleEsDao.save(articleEsDto);System.out.println(articleEsDto.getId());}

dev tools 查看

GET article/_search/

自定义方法测试

我们在上面创建接口的时候创建了一个根据标题查询的方法

    @Testvoid EsTest2(){List<ArticleEsDto> articleEsDtos = articleEsDao.findByTitle("青花瓷");System.out.println(articleEsDtos);}

 第二种方式

spring 默认提供了操作 ES 的客户端对象 ElasticSearchRestTemplate,同样提供了增删改查,更加灵活,适用于更加复杂的操作,返回结果更加完整,但是需要自己解析。

    @Resourceprivate ElasticsearchRestTemplate elasticsearchRestTemplate;

提示,在编写查询条件以及处理数据时,可以先在 Dev Tools 中执行一下查询,没问题之后再进行代码层面的条件编写。

查询 DSL

官方文档:Query and filter context | Elasticsearch Guide [8.14] | Elastic

查询模式:Boolean query | Elasticsearch Guide [8.14] | Elastic

GET /_search
{"query": {  "bool": {  //组合条件"must": [ //必须匹配{ "match":  // 模糊匹配{ "title":   "Search"        }},{ "match": { "content": "Elasticsearch" }}],"filter": [ //{ "term": //精确匹配 { "status": "published" }},{ "range": //范围匹配 { "publish_date": { "gte": "2015-01-01" }}}]}}
}

除了 must ,filter,还有 must_not,必须不存在才能匹配,should,至少有多少个条件相符才匹配,同时还有一个参数 minimum_should_match 满足最小的条件数,比如 1,至少满足一个条件才能查询到,比如标题和描述存在一个就可以返回结果。

POST _search
{"query": {"bool" : {"must" : {"term" : { "user.id" : "kimchy" }},"filter": {"term" : { "tags" : "production" }},"must_not" : {"range" : {"age" : { "gte" : 10, "lte" : 20 }}},"should" : [{ "term" : { "tags" : "env1" } },{ "term" : { "tags" : "deployed" } }],"minimum_should_match" : 1,"boost" : 1.0}}
}

需要注意的是,通过模糊查询之后,查询结果中有一个参数是 max_score,表示这条数据和搜索条件的最高匹配度。

在 Java 中编写查询条件以及处理查询的数据。

主要使用到的 API

//查询条件构造器
BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();//排序条件构造器
SortBuilder<?> sortBuilder = SortBuilders.scoreSort();//分页
PageRequest pageRequest = PageRequest.of((int) current, (int) pageSize);//组合查询条件
NativeSearchQuery searchQuery = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder).withPageable(pageRequest).withSorts(sortBuilder).build();
//调用 elasticsearchRestTemplate 查询
SearchHits<PostEsDTO> searchHits = elasticsearchRestTemplate.search(searchQuery, PostEsDTO.class);

完整代码

/*** 从 es 查询数据*/
@Service
public class ArticleEsManager {@Resourceprivate ArticleMapper articleMapper;@Resourceprivate ElasticsearchRestTemplate elasticsearchRestTemplate;public Page<Article> searchByEs(ArticleQueryRequest articleQueryRequest) {//提取查询参数Long id = articleQueryRequest.getId();String searchText = articleQueryRequest.getSearchText();String content = articleQueryRequest.getContent();String title = articleQueryRequest.getTitle();//设置分页参数,起始页为 0int current = articleQueryRequest.getCurrent() -1 ;int pageSize = articleQueryRequest.getPageSize();String sortField = articleQueryRequest.getSortField();String sortOrder = articleQueryRequest.getSortOrder();//构建布尔查询,创建 boolqueryBuilder,用于后续查询条件构建BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();//过滤查询条件//过滤掉被逻辑删除的 term 是精确匹配boolQueryBuilder.filter(QueryBuilders.termQuery("isDelete",0));//id必须精确匹配if(id!=null){boolQueryBuilder.filter(QueryBuilders.termQuery("id",id));}//模糊查询,按照查询词(关键词)检索if(StringUtils.isNotEmpty(searchText)){boolQueryBuilder.should(QueryBuilders.matchQuery("title",searchText));boolQueryBuilder.should(QueryBuilders.matchQuery("content",searchText));//设置至少多少匹配才进行查询boolQueryBuilder.minimumShouldMatch(1);}//根据标题检索if(StringUtils.isNotEmpty(title)){boolQueryBuilder.should(QueryBuilders.matchQuery("title",title));boolQueryBuilder.minimumShouldMatch(1);}//根据内容检索if(StringUtils.isNotEmpty(content)){boolQueryBuilder.should(QueryBuilders.matchQuery("content",searchText));//设置至少多少匹配才进行查询boolQueryBuilder.minimumShouldMatch(1);}//进行排序//对查询结果的分数进行排序SortBuilder<?> sortBuilder = SortBuilders.scoreSort();if (StringUtils.isNotEmpty(sortField) && StringUtils.isNotEmpty(sortOrder)){sortBuilder = SortBuilders.fieldSort(sortField);sortBuilder.order(CommonConstant.SORT_ORDER_ASC.equals(sortOrder) ? SortOrder.ASC:SortOrder.DESC);}//分页PageRequest pageRequest = PageRequest.of(current, pageSize);//构造查询NativeSearchQuery searchQuery = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder).withPageable(pageRequest).withSorts(sortBuilder).build();//调用 elasticsearchRestTemplate 执行查询SearchHits<ArticleEsDto> searchHits = elasticsearchRestTemplate.search(searchQuery, ArticleEsDto.class);System.out.println(searchHits);Page<Article> page = new Page<>();page.setTotal(searchHits.getTotalHits());//新建集合存储文章List<Article> resourceList = new ArrayList<>();//处理结果,判断是否有搜索结果if(searchHits.hasSearchHits()){//获取结果列表List<SearchHit<ArticleEsDto>> searchHits1 = searchHits.getSearchHits();System.out.println(searchHits1);//获取结果的id,使用id在数据库中查询List<Long> ids = searchHits1.stream().map(searchHit -> searchHit.getContent().getId()).collect(Collectors.toList());List<Article> articles = articleMapper.selectBatchIds(ids);//将查询结果与数据库查询结果进行匹配if (CollectionUtils.isNotEmpty(articles)) {//将查询结果与数据库查询结果进行匹配Map<Long, List<Article>> collect = articles.stream().collect(Collectors.groupingBy(Article::getId));//遍历文章idarticles.forEach(articleId -> {if(collect.containsKey(articleId)){resourceList.add(collect.get(articleId).get(0));}else{// 从 es 清空 db 已物理删除的数据String delete = elasticsearchRestTemplate.delete(String.valueOf(articleId), ArticleEsDto.class);}});}}page.setRecords(resourceList);return page;}
}

成功查询到数据 

完整代码

/*** 从 es 查询数据*/
@Service
public class ArticleEsManager {@Resourceprivate ArticleMapper articleMapper;@Resourceprivate ElasticsearchRestTemplate elasticsearchRestTemplate;public Page<Article> searchByEs(ArticleQueryRequest articleQueryRequest) {//提取查询参数Long id = articleQueryRequest.getId();String searchText = articleQueryRequest.getSearchText();String content = articleQueryRequest.getContent();String title = articleQueryRequest.getTitle();//设置分页参数,起始页为 0int current = articleQueryRequest.getCurrent() -1 ;int pageSize = articleQueryRequest.getPageSize();String sortField = articleQueryRequest.getSortField();String sortOrder = articleQueryRequest.getSortOrder();//构建布尔查询,创建 boolqueryBuilder,用于后续查询条件构建BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();//过滤查询条件//过滤掉被逻辑删除的 term 是精确匹配boolQueryBuilder.filter(QueryBuilders.termQuery("isDelete",0));//id必须精确匹配if(id!=null){boolQueryBuilder.filter(QueryBuilders.termQuery("id",id));}//模糊查询,按照查询词(关键词)检索if(StringUtils.isNotEmpty(searchText)){boolQueryBuilder.should(QueryBuilders.matchQuery("title",searchText));boolQueryBuilder.should(QueryBuilders.matchQuery("content",searchText));//设置至少多少匹配才进行查询boolQueryBuilder.minimumShouldMatch(1);}//根据标题检索if(StringUtils.isNotEmpty(title)){boolQueryBuilder.should(QueryBuilders.matchQuery("title",title));boolQueryBuilder.minimumShouldMatch(1);}//根据内容检索if(StringUtils.isNotEmpty(content)){boolQueryBuilder.should(QueryBuilders.matchQuery("content",searchText));//设置至少多少匹配才进行查询boolQueryBuilder.minimumShouldMatch(1);}//进行排序//对查询结果的分数进行排序SortBuilder<?> sortBuilder = SortBuilders.scoreSort();if (StringUtils.isNotEmpty(sortField) && StringUtils.isNotEmpty(sortOrder)){sortBuilder = SortBuilders.fieldSort(sortField);sortBuilder.order(CommonConstant.SORT_ORDER_ASC.equals(sortOrder) ? SortOrder.ASC:SortOrder.DESC);}//分页PageRequest pageRequest = PageRequest.of(current, pageSize);//构造查询NativeSearchQuery searchQuery = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder).withPageable(pageRequest).withSorts(sortBuilder).build();//调用 elasticsearchRestTemplate 执行查询SearchHits<ArticleEsDto> searchHits = elasticsearchRestTemplate.search(searchQuery, ArticleEsDto.class);System.out.println(searchHits);Page<Article> page = new Page<>();page.setTotal(searchHits.getTotalHits());//新建集合存储文章List<Article> resourceList = new ArrayList<>();//处理结果,判断是否有搜索结果if(searchHits.hasSearchHits()){//获取结果列表List<SearchHit<ArticleEsDto>> searchHits1 = searchHits.getSearchHits();System.out.println(searchHits1);//获取结果的id,使用id在数据库中查询List<Long> ids = searchHits1.stream().map(searchHit -> searchHit.getContent().getId()).collect(Collectors.toList());List<Article> articleList = articleMapper.selectBatchIds(ids);//将查询结果与数据库查询结果进行匹配if (CollectionUtils.isNotEmpty(articleList)) {//将查询结果与数据库查询结果进行匹配Map<Long, List<Article>> collect = articleList.stream().collect(Collectors.groupingBy(Article::getId));//遍历文章idids.forEach(articleId -> {if(collect.containsKey(articleId)){resourceList.add(collect.get(articleId).get(0));}else{// 从 es 清空 db 已物理删除的数据String delete = elasticsearchRestTemplate.delete(String.valueOf(articleId), ArticleEsDto.class);}});}}//设置到分页里page.setRecords(resourceList);return page;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/849949.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

React Hooks 封装可粘贴图片的输入框组件(wangeditor)

需求是需要一个文本框 但是可以支持右键或者ctrlv粘贴图片&#xff0c;原生js很麻烦&#xff0c;那不如用插件来实现吧~我这里用的wangeditor插件&#xff0c;初次写初次用&#xff0c;可能不太好&#xff0c;但目前是可以达到实现需求的一个效果啦&#xff01;后面再改进吧~ …

个位为0的数字-第13届蓝桥杯省赛Python真题精选

[导读]&#xff1a;超平老师的Scratch蓝桥杯真题解读系列在推出之后&#xff0c;受到了广大老师和家长的好评&#xff0c;非常感谢各位的认可和厚爱。作为回馈&#xff0c;超平老师计划推出《Python蓝桥杯真题解析100讲》&#xff0c;这是解读系列的第81讲。 个位为0的数字&am…

标准发布实施 |《新能源电池工业废水处理技术指南磷酸铁锂电池》

T/ACEF 130&#xff0d;2024《新能源电池工业废水处理技术指南磷酸铁锂电池》欢迎各单位引用执行&#xff01;有课题也可联合立项&#xff01; 发布日期&#xff1a;2024年02月04日 实施日期&#xff1a;2024年03月01日 主要起草人&#xff1a;刘愿军、孙冬、丁炜鹏、何小芬…

whistle手机抓包

环境&#xff1a;whistle&#xff1a;2.9.59 whistle手机抓包&#xff08;ios可以抓小程序的包&#xff1b;安卓机不能抓小程序的包&#xff0c;但是小程序的有开发者工具就够用了&#xff09; 以安卓手机为例&#xff08;手机跟电脑要连同一个wifi&#xff09; 1.电脑安装w…

Django项目上线-报错汇总

Django项目上线-报错汇总 下列报错基本都是Python环境相关 pip install 报错 WARNING: pip is configured with locations that require TLS/SSL, however the ssl module in Python is not available. debian运行pip报错ssl module in Python is not available - z417 - 博…

计算机专业本科论文起稿咋写

举例基于SpringBoot的Java基础的旅游管理系统 摘要 随着旅游业的快速发展&#xff0c;传统的旅游管理方式已经难以满足现代企业的需求。为了提高旅游企业的管理水平和服务质量&#xff0c;本文设计并实现了一个基于SpringBoot框架的旅游管理系统。本文首先介绍了旅游管理系统的…

Sql-labs的第一关

前言 我们在使用Sql-libs靶场进行Sql注入实验的时候&#xff0c;前提要求我们对mysql数据库结构要有一个大概的了解&#xff0c;因为mysql5.0以上的版本都会自带一个名为information_schema的数据库&#xff0c;这个数据库下面会有columns和tables两个表。 tables这个表的table…

3-1RT-Thread时钟管理

这里写自定义目录标题 时钟节拍是RT thread操作系统的最小时间单位。 第一个功能&#xff0c;rt tick值自动加1&#xff0c;在RT thread当中通过RT_USING_SMP定义了多核和单核的场景。第二个功能&#xff0c;检查当前线程的时间片&#xff0c;首先获取当前线程&#xff0c;将当…

Vitis HLS 学习笔记--初始化与复位

目录 1. 简介 2. 控制初始化与复位 2.1 初始化 2.2 复位 2.3 全局复位选项 2.4 复位排除 3. 阵列初始化和复位 3.1 不使用 static 限定符 3.2 使用 static 限定符 3.3 BRAM 和 URAM 4. 总结 1. 简介 本文对比分析两个方面的初始化和复位&#xff1a;阵列和控制&…

Java采取擦除式泛型到底兼容了什么场景?

在开始前刚好我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「 Java的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01;Java擦除式泛型是一个妥协,…

美颜相机「BeautyCam」v12.0.80 祛广告解索会员版(美妆相机功能,展现女神魅力)

软件介绍 美颜相机&#xff0c;一款由知名移动互联网企业Meitu Inc.开发的移动设备照片编辑与美化应用&#xff0c;起初主要针对娱乐消费市场&#xff0c;随后集成了商业营销功能。目前&#xff0c;它已跻身全球最受欢迎的手机摄影应用程序之列。在中国&#xff0c;美颜相机和…

JAVA高级-反射与动态代理(十五)

观看尚硅谷视频做的视频笔记 一、高级-反射与动态代理&#xff08;十四&#xff09; 1、反射概述 1&#xff09;java程序中&#xff0c;所有的对象都有两种类型&#xff1a;编译时类型&#xff0c;运行时类型&#xff0c;而很多时候对象的编译类型和运行时类型不一致。 此处…

workerman error 2 send buffer full and drop package

来源 报错信息&#xff1a;workerman error 2 send buffer full and drop package 定时发送数据的时候&#xff0c;本地偶尔出现这种情况 线上第一条数据发出去就报错了&#xff0c;数据改小一点可以发&#xff0c;不过一会还是会出现这种情况。 解决 根据我的经验&#xf…

SAP HCM OPT函数作用

导读 INTRODUCTION OPT函数&#xff1a;SAP HCM工资核算是很多函数的汇总集&#xff0c;原有有兴趣问过SAP的人为什么SCHEMA需要这样设计&#xff0c;SAP的人说是用汇编的逻辑设计的&#xff0c;当时是尽可能用机器语言加速速度读取&#xff0c;每个函数都有对应的业务逻辑代码…

React实战(一)初始化项目、配置router、redux、axios

(一)初始化项目 1.安装项目 npx create-react-app 项目名 编译报错&#xff1a; 解决办法&#xff1a;安装最新的babel-preset-react-app npm install babel-preset-react-applatest 2.配置项目 (1)配置文件目录 (2)使用craco配置webpack.config npm install craco/crac…

Coze入门指南:创建Bot时,如何写好人设与回复逻辑(Persona Prompt)

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 Coze Bot 📒📝 Persona & Prompt🌟 # Character🌟 ## Skills🌟 # Overall Rules to follow🌟 ## Workflow🌟 ## Constraints📝 通用写法与模板📝 示例🌟技巧和注意事项⚓️ 相关链接 ⚓️📖 介绍 📖…

免费使用GPT-4生成图片的方法

写在前言 hello&#xff0c;大家好&#xff0c;我是一点&#xff0c;喜欢编程&#xff0c;目前持续在关注AI领域的发展&#xff0c;希望可以结交一些有意思的朋友。 大家可以关注我的公号&#xff1a;【一点sir】&#xff0c;了解更多文章&#xff0c;希望可以和你们成为朋友…

OpenCompass 大模型评测平台C-Eval 基准任务评估实战

1. 引言 在人工智能迅速发展的今天&#xff0c;大型语言模型&#xff08;LLMs&#xff09;在多个领域展现出了巨大的潜力和应用价值。然而&#xff0c;如何评价这些模型的性能&#xff0c;了解它们的优缺点&#xff0c;成为了一个重要课题。OpenCompass&#xff0c;一个由上海…

使用Cython编译Python源码加密加速,有这一篇就够了!

0 前言 python是一门脚本语言&#xff0c;运行时由python虚拟机解释执行。当我们使用python设计好算法给第三方使用时只能提供源码&#xff0c;任何运行我们算法的人都可以看到源码以及对应的算法思路。因此&#xff0c;需要一定手动保护源码。 最简单的保护方式是使用代码混…

高光谱图像聚类的像素-超像素对比学习与伪标签校正

Pixel-Superpixel Contrastive Learning and Pseudo-Label Correction for Hyperspectral Image Clustering 文章目录 Pixel-Superpixel Contrastive Learning and Pseudo-Label Correction for Hyperspectral Image Clustering摘要引言相关方法对比学习 方法超像素对比学习像素…