SpringBoot+ElasticSearch 实现模糊查询,批量CRUD,排序,分页,高亮!

分手后的思念是犯贱 2023-10-03 22:23 64阅读 0赞

一、导入elasticsearch依赖

在pom.xml里加入如下依赖

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-data-elasticsearch</artifactId>
  4. </dependency>
  5. 复制代码

非常重要:检查依赖版本是否与你当前所用的版本是否一致,如果不一致,会连接失败!

二、创建高级客户端

  1. import org.apache.http.HttpHost;
  2. import org.elasticsearch.client.RestClient;
  3. import org.elasticsearch.client.RestHighLevelClient;
  4. import org.springframework.context.annotation.Bean;
  5. import org.springframework.context.annotation.Configuration;
  6. @Configuration
  7. public class ElasticSearchClientConfig {
  8. @Bean
  9. public RestHighLevelClient restHighLevelClient(){
  10. RestHighLevelClient client = new RestHighLevelClient(
  11. RestClient.builder(
  12. new HttpHost("服务器IP", 9200, "http")));
  13. return client;
  14. }
  15. }
  16. 复制代码

三、基本用法

1.创建、判断存在、删除索引

  1. import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequest;
  2. import org.elasticsearch.action.support.master.AcknowledgedResponse;
  3. import org.elasticsearch.client.RequestOptions;
  4. import org.elasticsearch.client.RestHighLevelClient;
  5. import org.elasticsearch.client.indices.CreateIndexRequest;
  6. import org.elasticsearch.client.indices.CreateIndexResponse;
  7. import org.elasticsearch.client.indices.GetIndexRequest;
  8. import org.junit.jupiter.api.Test;
  9. import org.springframework.beans.factory.annotation.Autowired;
  10. import org.springframework.boot.test.context.SpringBootTest;
  11. import java.io.IOException;
  12. @SpringBootTest
  13. class ElasticsearchApplicationTests {
  14. @Autowired
  15. private RestHighLevelClient restHighLevelClient;
  16. @Test
  17. void testCreateIndex() throws IOException {
  18. //1.创建索引请求
  19. CreateIndexRequest request = new CreateIndexRequest("ljx666");
  20. //2.客户端执行请求IndicesClient,执行create方法创建索引,请求后获得响应
  21. CreateIndexResponse response=
  22. restHighLevelClient.indices().create(request, RequestOptions.DEFAULT);
  23. System.out.println(response);
  24. }
  25. @Test
  26. void testExistIndex() throws IOException {
  27. //1.查询索引请求
  28. GetIndexRequest request=new GetIndexRequest("ljx666");
  29. //2.执行exists方法判断是否存在
  30. boolean exists=restHighLevelClient.indices().exists(request,RequestOptions.DEFAULT);
  31. System.out.println(exists);
  32. }
  33. @Test
  34. void testDeleteIndex() throws IOException {
  35. //1.删除索引请求
  36. DeleteIndexRequest request=new DeleteIndexRequest("ljx666");
  37. //执行delete方法删除指定索引
  38. AcknowledgedResponse delete = restHighLevelClient.indices().delete(request, RequestOptions.DEFAULT);
  39. System.out.println(delete.isAcknowledged());
  40. }
  41. }
  42. 复制代码

2.对文档的CRUD

创建文档:

注意:如果添加时不指定文档ID,他就会随机生成一个ID,ID唯一。

创建文档时若该ID已存在,发送创建文档请求后会更新文档中的数据。

  1. @Test
  2. void testAddUser() throws IOException {
  3. //1.创建对象
  4. User user=new User("Go",21,new String[]{"内卷","吃饭"});
  5. //2.创建请求
  6. IndexRequest request=new IndexRequest("ljx666");
  7. //3.设置规则 PUT /ljx666/_doc/1
  8. //设置文档id=6,设置超时=1s等,不设置会使用默认的
  9. //同时支持链式编程如 request.id("6").timeout("1s");
  10. request.id("6");
  11. request.timeout("1s");
  12. //4.将数据放入请求,要将对象转化为json格式
  13. //XContentType.JSON,告诉它传的数据是JSON类型
  14. request.source(JSONValue.toJSONString(user), XContentType.JSON);
  15. //5.客户端发送请求,获取响应结果
  16. IndexResponse indexResponse=restHighLevelClient.index(request,RequestOptions.DEFAULT);
  17. System.out.println(indexResponse.toString());
  18. System.out.println(indexResponse.status());
  19. }
  20. 复制代码

获取文档中的数据:

  1. @Test
  2. void testGetUser() throws IOException {
  3. //1.创建请求,指定索引、文档id
  4. GetRequest request=new GetRequest("ljx666","1");
  5. GetResponse getResponse=restHighLevelClient.get(request,RequestOptions.DEFAULT);
  6. System.out.println(getResponse);//获取响应结果
  7. //getResponse.getSource() 返回的是Map集合
  8. System.out.println(getResponse.getSourceAsString());//获取响应结果source中内容,转化为字符串
  9. }
  10. 复制代码

更新文档数据:

注意:需要将User对象中的属性全部指定值,不然会被设置为空,如User只设置了名称,那么只有名称会被修改成功,其他会被修改为null。

  1. @Test
  2. void testUpdateUser() throws IOException {
  3. //1.创建请求,指定索引、文档id
  4. UpdateRequest request=new UpdateRequest("ljx666","6");
  5. User user =new User("GoGo",21,new String[]{"内卷","吃饭"});
  6. //将创建的对象放入文档中
  7. request.doc(JSONValue.toJSONString(user),XContentType.JSON);
  8. UpdateResponse updateResponse=restHighLevelClient.update(request,RequestOptions.DEFAULT);
  9. System.out.println(updateResponse.status());//更新成功返回OK
  10. }
  11. 复制代码

删除文档:

  1. @Test
  2. void testDeleteUser() throws IOException {
  3. //创建删除请求,指定要删除的索引与文档ID
  4. DeleteRequest request=new DeleteRequest("ljx666","6");
  5. DeleteResponse updateResponse=restHighLevelClient.delete(request,RequestOptions.DEFAULT);
  6. System.out.println(updateResponse.status());//删除成功返回OK,没有找到返回NOT_FOUND
  7. }
  8. 复制代码

3.批量CRUD数据

这里只列出了批量插入数据,其他与此类似

注意:hasFailures()方法是返回是否失败,即它的值为false时说明上传成功

  1. @Test
  2. void testBulkAddUser() throws IOException {
  3. BulkRequest bulkRequest=new BulkRequest();
  4. //设置超时
  5. bulkRequest.timeout("10s");
  6. ArrayList<User> list=new ArrayList<>();
  7. list.add(new User("Java",25,new String[]{"内卷"}));
  8. list.add(new User("Go",18,new String[]{"内卷"}));
  9. list.add(new User("C",30,new String[]{"内卷"}));
  10. list.add(new User("C++",26,new String[]{"内卷"}));
  11. list.add(new User("Python",20,new String[]{"内卷"}));
  12. int id=1;
  13. //批量处理请求
  14. for (User u :list){
  15. //不设置id会生成随机id
  16. bulkRequest.add(new IndexRequest("ljx666")
  17. .id(""+(id++))
  18. .source(JSONValue.toJSONString(u),XContentType.JSON));
  19. }
  20. BulkResponse bulkResponse=restHighLevelClient.bulk(bulkRequest,RequestOptions.DEFAULT);
  21. System.out.println(bulkResponse.hasFailures());//是否执行失败,false为执行成功
  22. }
  23. 复制代码

4.查询所有、模糊查询、分页查询、排序、高亮显示

  1. @Test
  2. void testSearch() throws IOException {
  3. SearchRequest searchRequest=new SearchRequest("ljx666");//里面可以放多个索引
  4. SearchSourceBuilder sourceBuilder=new SearchSourceBuilder();//构造搜索条件
  5. //此处可以使用QueryBuilders工具类中的方法
  6. //1.查询所有
  7. sourceBuilder.query(QueryBuilders.matchAllQuery());
  8. //2.查询name中含有Java的
  9. sourceBuilder.query(QueryBuilders.multiMatchQuery("java","name"));
  10. //3.分页查询
  11. sourceBuilder.from(0).size(5);
  12. //4.按照score正序排列
  13. //sourceBuilder.sort(SortBuilders.scoreSort().order(SortOrder.ASC));
  14. //5.按照id倒序排列(score会失效返回NaN)
  15. //sourceBuilder.sort(SortBuilders.fieldSort("_id").order(SortOrder.DESC));
  16. //6.给指定字段加上指定高亮样式
  17. HighlightBuilder highlightBuilder=new HighlightBuilder();
  18. highlightBuilder.field("name").preTags("<span style='color:red;'>").postTags("</span>");
  19. sourceBuilder.highlighter(highlightBuilder);
  20. searchRequest.source(sourceBuilder);
  21. SearchResponse searchResponse=restHighLevelClient.search(searchRequest,RequestOptions.DEFAULT);
  22. //获取总条数
  23. System.out.println(searchResponse.getHits().getTotalHits().value);
  24. //输出结果数据(如果不设置返回条数,大于10条默认只返回10条)
  25. SearchHit[] hits=searchResponse.getHits().getHits();
  26. for(SearchHit hit :hits){
  27. System.out.println("分数:"+hit.getScore());
  28. Map<String,Object> source=hit.getSourceAsMap();
  29. System.out.println("index->"+hit.getIndex());
  30. System.out.println("id->"+hit.getId());
  31. for(Map.Entry<String,Object> s:source.entrySet()){
  32. System.out.println(s.getKey()+"--"+s.getValue());
  33. }
  34. }
  35. }
  36. 复制代码

四、总结

1.大致流程

创建对应的请求 —> 设置请求(添加规则,添加数据等) —> 执行对应的方法(传入请求,默认请求选项)–> 接收响应结果(执行方法返回值)–> 输出响应结果中需要的数据(source,status等)

2.注意事项

  • 如果不指定id,会自动生成一个随机id
  • 正常情况下,不应该这样使用new IndexRequest(“ljx777”),如果索引发生改变了,那么代码都需要修改,可以定义一个枚举类或者一个专门存放常量的类,将变量用final static等进行修饰,并指定索引值。其他地方引用该常量即可,需要修改也只需修改该类即可。
  • elasticsearch相关的东西,版本都必须一致,不然会报错
  • elasticsearch很消耗内存,建议在内存较大的服务器上运行elasticsearch,否则会因为内存不足导致elasticsearch自动killed

发表评论

表情:
评论列表 (有 0 条评论,64人围观)

还没有评论,来说两句吧...

相关阅读