- 資訊首頁(yè) > 開(kāi)發(fā)技術(shù) > 編程語(yǔ)言 >
- springboot中使用ElasticSearch的詳細教程
新建一個(gè)springboot項目springboot_es
用于本次與ElasticSearch的整合,如下圖
修改我們的pom.xml
,加入spring-boot-starter-data-elasticsearch
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-elasticsearch</artifactId> </dependency>
由于ElasticSearch從7.x版本開(kāi)始淡化TransportClient甚至于在8.x版本中遺棄,所以spring data elasticsearch推薦我們使用rest客戶(hù)端RestHingLevelClient
(端口號使用9200)以及接口ElasticSearchRespositoy
。
首先我們編寫(xiě)配置文件如下
/** * ElasticSearch Rest Client config * @author Christy * @date 2021/4/29 19:40 **/ @Configuration public class ElasticSearchRestClientConfig extends AbstractElasticsearchConfiguration{ @Override @Bean public RestHighLevelClient elasticsearchClient() { final ClientConfiguration clientConfiguration = ClientConfiguration.builder() .connectedTo("192.168.8.101:9200") .build(); return RestClients.create(clientConfiguration).rest(); } }
有了上面的rest client,我們就可以在其他的地方注入該客戶(hù)端對ElasticSearch進(jìn)行操作。我們新建一個(gè)測試文件,使用客戶(hù)端對ElasticSearch進(jìn)行基本的操作
1.注入RestClient
/** * ElasticSearch Rest client操作 * * RestHighLevelClient 更強大,更靈活,但是不能友好的操作對象 * ElasticSearchRepository 對象操作友好 * * 我們使用rest client 主要測試文檔的操作 * @Author Christy * @Date 2021/4/29 19:51 **/ @SpringBootTest public class TestRestClient { // 復雜查詢(xún)使用:比如高亮查詢(xún) @Autowired private RestHighLevelClient restHighLevelClient; }
2.插入一條文檔
/** * 新增一條文檔 * @author Christy * @date 2021/4/29 20:17 */ @Test public void testAdd() throws IOException { /** * 向ES中的索引christy下的type類(lèi)型中添加一天文檔 */ IndexRequest indexRequest = new IndexRequest("christy","user","11"); indexRequest.source("{\"name\":\"齊天大圣孫悟空\",\"age\":685,\"bir\":\"1685-01-01\",\"introduce\":\"花果山水簾洞美猴王齊天大圣孫悟空是也!\"," + "\"address\":\"花果山\"}", XContentType.JSON); IndexResponse indexResponse = restHighLevelClient.index(indexRequest, RequestOptions.DEFAULT); System.out.println(indexResponse.status()); }
我們可以看到文檔插入成功,我們去kibana中查詢(xún)該條文檔
完全沒(méi)有問(wèn)題的。
3.刪除一條文檔
/** * 刪除一條文檔 * @author Christy * @date 2021/4/29 20:18 */ @Test public void deleteDoc() throws IOException { // 我們把特朗普刪除了 DeleteRequest deleteRequest = new DeleteRequest("christy","user","rYBNG3kBRz-Sn-2f3ViU"); DeleteResponse deleteResponse = restHighLevelClient.delete(deleteRequest, RequestOptions.DEFAULT); System.out.println(deleteResponse.status()); } }
4.更新一條文檔
/** * 更新一條文檔 * @author Christy * @date 2021/4/29 20:19 */ @Test public void updateDoc() throws IOException { UpdateRequest updateRequest = new UpdateRequest("christy","user","p4AtG3kBRz-Sn-2fMFjj"); updateRequest.doc("{\"name\":\"調皮搗蛋的hardy\"}",XContentType.JSON); UpdateResponse updateResponse = restHighLevelClient.update(updateRequest, RequestOptions.DEFAULT); System.out.println(updateResponse.status()); }
5.批量更新文檔
/** * 批量更新 * @author Christy * @date 2021/4/29 20:42 */ @Test public void bulkUpdate() throws IOException { BulkRequest bulkRequest = new BulkRequest(); // 添加 IndexRequest indexRequest = new IndexRequest("christy","user","13"); indexRequest.source("{\"name\":\"天蓬元帥豬八戒\",\"age\":985,\"bir\":\"1685-01-01\",\"introduce\":\"天蓬元帥豬八戒因調戲嫦娥被貶下凡\",\"address\":\"高老莊\"}", XContentType.JSON); bulkRequest.add(indexRequest); // 刪除 DeleteRequest deleteRequest01 = new DeleteRequest("christy","user","pYAtG3kBRz-Sn-2fMFjj"); DeleteRequest deleteRequest02 = new DeleteRequest("christy","user","uhTyGHkBExaVQsl4F9Lj"); DeleteRequest deleteRequest03 = new DeleteRequest("christy","user","C8zCGHkB5KgTrUTeLyE_"); bulkRequest.add(deleteRequest01); bulkRequest.add(deleteRequest02); bulkRequest.add(deleteRequest03); // 修改 UpdateRequest updateRequest = new UpdateRequest("christy","user","10"); updateRequest.doc("{\"name\":\"煉石補天的女?huà)z\"}",XContentType.JSON); bulkRequest.add(updateRequest); BulkResponse bulkResponse = restHighLevelClient.bulk(bulkRequest, RequestOptions.DEFAULT); BulkItemResponse[] items = bulkResponse.getItems(); for (BulkItemResponse item : items) { System.out.println(item.status()); } }
在kibana中查詢(xún)結果
6.查詢(xún)文檔
@Test public void testSearch() throws IOException { //創(chuàng )建搜索對象 SearchRequest searchRequest = new SearchRequest("christy"); //搜索構建對象 SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder(); searchSourceBuilder.query(QueryBuilders.matchAllQuery())//執行查詢(xún)條件 .from(0)//起始條數 .size(10)//每頁(yè)展示記錄 .postFilter(QueryBuilders.matchAllQuery()) //過(guò)濾條件 .sort("age", SortOrder.DESC);//排序 //創(chuàng )建搜索請求 searchRequest.types("user").source(searchSourceBuilder); SearchResponse searchResponse = restHighLevelClient.search(searchRequest, RequestOptions.DEFAULT); System.out.println("符合條件的文檔總數: "+searchResponse.getHits().getTotalHits()); System.out.println("符合條件的文檔最大得分: "+searchResponse.getHits().getMaxScore()); SearchHit[] hits = searchResponse.getHits().getHits(); for (SearchHit hit : hits) { System.out.println(hit.getSourceAsMap()); } }
1.準備工作
ElasticSearchRepository方式主要通過(guò)注解和對接口實(shí)現的方式來(lái)實(shí)現ES的操作,我們在實(shí)體類(lèi)上通過(guò)注解配置ES索引的映射關(guān)系后,當實(shí)現了ElasticSearchRepository接口的類(lèi)第一次操作ES進(jìn)行插入文檔的時(shí)候,ES會(huì )自動(dòng)生成所需要的一切。但是該種方式無(wú)法實(shí)現高亮查詢(xún),想要實(shí)現高亮查詢(xún)只能使用RestHighLevelClient
開(kāi)始之前我們需要熟悉一下接口方式為我們提供的注解,以及編寫(xiě)一些基礎的類(lèi)
1.清空ES數據
2.了解注解
@Document
: 代表一個(gè)文檔記錄
indexName
: 用來(lái)指定索引名稱(chēng)
type
: 用來(lái)指定索引類(lèi)型
@Id
: 用來(lái)將對象中id和ES中_id映射
@Field
: 用來(lái)指定ES中的字段對應Mapping
type
: 用來(lái)指定ES中存儲類(lèi)型
analyzer
: 用來(lái)指定使用哪種分詞器
3.新建實(shí)體類(lèi)
/** * 用在類(lèi)上作用:將Emp的對象映射成ES中一條json格式文檔 * indexName: 用來(lái)指定這個(gè)對象的轉為json文檔存入那個(gè)索引中 要求:ES服務(wù)器中之前不能存在此索引名 * type : 用來(lái)指定在當前這個(gè)索引下創(chuàng )建的類(lèi)型名稱(chēng) * * @Author Christy * @Date 2021/4/29 21:22 */ @Data @Document(indexName = "christy",type = "user") public class User { @Id //用來(lái)將對象中id屬性與文檔中_id 一一對應 private String id; // 用在屬性上 代表mapping中一個(gè)屬性 一個(gè)字段 type:屬性 用來(lái)指定字段類(lèi)型 analyzer:指定分詞器 @Field(type = FieldType.Text,analyzer = "ik_max_word") private String name; @Field(type = FieldType.Integer) private Integer age; @Field(type = FieldType.Date) @JsonFormat(pattern = "yyyy-MM-dd") private Date bir; @Field(type = FieldType.Text,analyzer = "ik_max_word") private String content; @Field(type = FieldType.Text,analyzer = "ik_max_word") private String address; }
4.UserRepository
/** * @Author Christy * @Date 2021/4/29 21:23 **/ public interface extends ElasticsearchRepository<User,String> { }
5.TestUserRepository
/** * @Author Christy * @Date 2021/4/29 21:51 **/ @SpringBootTest public class TestUserRepository { @Autowired private UserRepository userRepository; }
2.保存文檔
@Test public void testSaveAndUpdate(){ User user = new User(); // id初識為空,此操作為新增 user.setId(UUID.randomUUID().toString()); user.setName("唐三藏"); user.setBir(new Date()); user.setIntroduce("西方世界如來(lái)佛祖大弟子金蟬子轉世,十世修行的好人,得道高僧!"); user.setAddress("大唐白馬寺"); userRepository.save(user); }
3.修改文檔
@Test public void testSaveAndUpdate(){ User user = new User(); // 根據id修改信息 user.setId("1666eb47-0bbf-468b-ab45-07758c741461"); user.setName("唐三藏"); user.setBir(new Date()); user.setIntroduce("俗家姓陳,狀元之后。西方世界如來(lái)佛祖大弟子金蟬子轉世,十世修行的好人,得道高僧!"); user.setAddress("大唐白馬寺"); userRepository.save(user); }
4.刪除文檔
repository接口默認提供了4種刪除方式,我們演示根據id進(jìn)行刪除
@Test public void deleteDoc(){ userRepository.deleteById("1666eb47-0bbf-468b-ab45-07758c741461"); }
5.檢索一條記錄
@Test public void testFindOne(){ Optional<User> optional = userRepository.findById("1666eb47-0bbf-468b-ab45-07758c741461"); System.out.println(optional.get()); }
6.查詢(xún)所有
@Test public void testFindAll(){ Iterable<User> all = userRepository.findAll(); all.forEach(user-> System.out.println(user)); }
7.排序
@Test public void testFindAllSort(){ Iterable<User> all = userRepository.findAll(Sort.by(Sort.Order.asc("age"))); all.forEach(user-> System.out.println(user)); }
8.分頁(yè)
@Test public void testFindPage(){ //PageRequest.of 參數1: 當前頁(yè)-1 Page<User> search = userRepository.search(QueryBuilders.matchAllQuery(), PageRequest.of(1, 1)); search.forEach(user-> System.out.println(user)); }
9.自定義查詢(xún)
先給大家看一個(gè)表,是不是很暈_(¦3」∠)_
這個(gè)表格看起來(lái)復雜,實(shí)際上也不簡(jiǎn)單,但是確實(shí)很牛逼。我們只要按照上面的定義在接口中定義相應的方法,無(wú)須寫(xiě)實(shí)現就可實(shí)現我們想要的功能
舉個(gè)例子,上面有個(gè)findByName是下面這樣定義的
假如我們現在有個(gè)需求需要按照名字查詢(xún)用戶(hù),我們可以在UserRepository
中定義一個(gè)方法,如下
// 根據姓名查詢(xún) List<User> findByName(String name);
系統提供的查詢(xún)方法中findBy
是一個(gè)固定寫(xiě)法,像上面我們定義的方法findByName
,其中Name是我們實(shí)體類(lèi)中的屬性名,這個(gè)必須對應上。也就是說(shuō)這個(gè)findByName不僅僅局限于name
,還可以findByAddress、findByAge等等;
現在就拿findByName來(lái)講,我們要查詢(xún)名字叫唐三藏的用戶(hù)
@Test public void testFindByName(){ List<User> userList = userRepository.findByName("唐三藏"); userList.forEach(user-> System.out.println(user)); }
其實(shí)就是框架底層直接使用下面的命令幫我們實(shí)現的查詢(xún)
GET /christy/user/_search { "query": { "bool": { "must": [ { "term": { "name":"?" } } ] } } }
10.高亮查詢(xún)
我們上面說(shuō)了,ElasticSearchRepository實(shí)現不了高亮查詢(xún),想要實(shí)現高亮查詢(xún)還是需要使用RestHighLevelClient方式。最后我們使用rest clientl實(shí)現一次高亮查詢(xún)
@Test public void testHighLightQuery() throws IOException, ParseException { // 創(chuàng )建搜索請求 SearchRequest searchRequest = new SearchRequest("christy"); // 創(chuàng )建搜索對象 SearchSourceBuilder searchSourceBuilder = new SearchSourceBuilder(); searchSourceBuilder.query(QueryBuilders.termQuery("introduce", "唐僧")) // 設置查詢(xún)條件 .from(0) // 起始條數(當前頁(yè)-1)*size的值 .size(10) // 每頁(yè)展示條數 .sort("age", SortOrder.DESC) // 排序 .highlighter(new HighlightBuilder().field("*").requireFieldMatch(false).preTags("<span style='color:red;'>").postTags("</span>")); // 設置高亮 searchRequest.types("user").source(searchSourceBuilder); SearchResponse searchResponse = restHighLevelClient.search(searchRequest, RequestOptions.DEFAULT); SearchHit[] hits = searchResponse.getHits().getHits(); List<User> userList = new ArrayList<>(); for (SearchHit hit : hits) { Map<String, Object> sourceAsMap = hit.getSourceAsMap(); User user = new User(); user.setId(hit.getId()); user.setAge(Integer.parseInt(sourceAsMap.get("age").toString())); user.setBir(new SimpleDateFormat("yyyy-MM-dd").parse(sourceAsMap.get("bir").toString())); user.setIntroduce(sourceAsMap.get("introduce").toString()); user.setName(sourceAsMap.get("name").toString()); user.setAddress(sourceAsMap.get("address").toString()); Map<String, HighlightField> highlightFields = hit.getHighlightFields(); if(highlightFields.containsKey("name")){ user.setName(highlightFields.get("name").fragments()[0].toString()); } if(highlightFields.containsKey("introduce")){ user.setIntroduce(highlightFields.get("introduce").fragments()[0].toString()); } if(highlightFields.containsKey("address")){ user.setAddress(highlightFields.get("address").fragments()[0].toString()); } userList.add(user); } userList.forEach(user -> System.out.println(user)); }
到此這篇關(guān)于ElasticSearch在springboot中使用的詳細教程的文章就介紹到這了,更多相關(guān)springboot使用ElasticSearch內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng )、來(lái)自本網(wǎng)站內容采集于網(wǎng)絡(luò )互聯(lián)網(wǎng)轉載等其它媒體和分享為主,內容觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如侵犯了原作者的版權,請告知一經(jīng)查實(shí),將立刻刪除涉嫌侵權內容,聯(lián)系我們QQ:712375056,同時(shí)歡迎投稿傳遞力量。
Copyright ? 2009-2022 56dr.com. All Rights Reserved. 特網(wǎng)科技 特網(wǎng)云 版權所有 特網(wǎng)科技 粵ICP備16109289號
域名注冊服務(wù)機構:阿里云計算有限公司(萬(wàn)網(wǎng)) 域名服務(wù)機構:煙臺帝思普網(wǎng)絡(luò )科技有限公司(DNSPod) CDN服務(wù):阿里云計算有限公司 百度云 中國互聯(lián)網(wǎng)舉報中心 增值電信業(yè)務(wù)經(jīng)營(yíng)許可證B2
建議您使用Chrome、Firefox、Edge、IE10及以上版本和360等主流瀏覽器瀏覽本網(wǎng)站