Spring集成ElasticSearch搜索引擎

[toc]

ElasticSearch是一个基于Lucene的搜索服务器。它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口。Elasticsearch是用Java开发的,并作为Apache许可条款下的开放源码发布,是当前流行的企业级搜索引擎。设计用于云计算中,能够达到实时搜索,稳定,可靠,快速,安装使用方便。

我们建立一个网站或应用程序,并要添加搜索功能,但是想要完成搜索工作的创建是非常困难的。我们希望搜索解决方案要运行速度快,我们希望能有一个零配置和一个完全免费的搜索模式,我们希望能够简单地使用JSON通过HTTP来索引数据,我们希望我们的搜索服务器始终可用,我们希望能够从一台开始并扩展到数百台,我们要实时搜索,我们要简单的多租户,我们希望建立一个云的解决方案。因此我们利用Elasticsearch来解决所有这些问题以及可能出现的更多其它问题。

我们需要将这中搜索模式集中到我们的项目中去,从而实现大量数据的搜索功能,性能远远超越数据库查询!

本文引用燕归来http://www.zhoutao123.com/?p=95

ElasticSearch : You Know, for Search!

前期安装

这里安装不在过多的赘述,这里推荐官方的中文文档,安装指引 安装完成之后,运行启动,打开浏览器,访问http://localhost:9200 将看到一下信息(cluster_name非常重要后面的代码要用到):

{
  "name" : "2g-ucq2",
  "cluster_name" : "elasticsearch",
  "cluster_uuid" : "tpMpsSenRpaBPugKlG7QaA",
  "version" : {
    "number" : "5.5.0",
    "build_hash" : "260387d",
    "build_date" : "2017-06-30T23:16:05.735Z",
    "build_snapshot" : false,
    "lucene_version" : "6.6.0"
  },
  "tagline" : "You Know, for Search"
}

Maven支持库安装

我们使用Maven,因此需要修改POM文件内容,添加如下依赖即可。

        <!--ElasticSearch支持文件-->
        <dependency>
            <groupId>org.elasticsearch.client</groupId>
            <artifactId>transport</artifactId>
            <version>5.5.0</version>
        </dependency>


        <dependency>
            <groupId>org.apache.logging.log4j</groupId>
            <artifactId>log4j-core</artifactId>
            <version>2.7</version>
        </dependency>

添加log4j的配置文件

在classpath源目录下创建log4j2.properties文件,内容信息如下

appender.console.type =Console
appender.console.name =Console
appender.console.layout.type =PatternLayout
appender.console.layout.pattern = [%t]  %-5p %c  - %m%n

rootLogger.lever = info
rootLogger.appenderRef.console.ref = Console

创建Client客户端

我们将核心的client对象,使用配置注解进行配置,方便日后取用

package com.tao.spring.config;

import org.elasticsearch.client.transport.TransportClient;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.transport.InetSocketTransportAddress;
import org.elasticsearch.transport.client.PreBuiltTransportClient;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import java.net.InetAddress;
import java.net.UnknownHostException;

/**
 * 本文件由周涛创建,位于com.tao.spring.config包下
 * 创建时间2018/1/26 20:42
 * 邮箱:zhoutao@xiaodouwangluo.com
 * 作用:创建ES的client对象,方便后期使用
 */

@Configuration
public class ElasticSearchConfig {

    @Bean
    public TransportClient client() throws UnknownHostException{
        //注意这里的端口是TCP端口9300,而非HTTP接口9200
        InetSocketTransportAddress node = new InetSocketTransportAddress(InetAddress.getByName("localhost"),9300);
        //机器名称,可以首页查询,这个不能出现错误,否则无法连接ES
        Settings settings=Settings.builder().put("cluster.name","elasticsearch").build();
        TransportClient client = new PreBuiltTransportClient( settings);
         client.addTransportAddress(node);
         return client;
    }
}

实现增删改查以及符合查询

下面将依次实现数据的增删改查,以及符合查询,这里我们的的类型是book,其具有两个属性分别是bookid和bookprice,具体说明参考注释。
首先需要注入对象

    @Autowired
     TransportClient client;

实现查询数据

需求:实现根据bookid获取数据的信息

/**
     * 使用ES进行简单的搜索操作
     * @param bookid
     * @return
     */
    @RequestMapping(value = "/get",method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
    @ResponseBody
    public String getInfoById(@RequestParam(name = "bookid") String bookid){
        if (Strings.isNullOrEmpty(bookid)){
            return "缺少参数,或参数格式不正确";
        }
        //tao指定的书索引,book指的是类型,后面相同,不在做叙述
        GetResponse getFields = client.prepareGet("tao", "book", bookid).get();
        //这里需要做些判断,没有没有判断,在没有查到的情况,会抛出异常信息
        if ( getFields ==null || !getFields.isExists()){
            return "没有查询到结果信息";
        }else{
            return getFields.getSource().toString();
        }
    }

实现添加数据

需求:提供bookid和bookprice来插入数据到ES

/**
     * 插入一条数据到ES中
     * @param bookid
     * @param bookprice
     * @return
     **/
    @RequestMapping(value = "/add",method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
    @ResponseBody
    public String addInfoToES(String bookid,float bookprice){
        if (Strings.isNullOrEmpty(bookid) || bookprice <= 0){
            return "参数不存在或者参数格式异常!";
        }

        try {
        //构造JSON数据,方便使用
            XContentBuilder xContentBuilder = XContentFactory.jsonBuilder().startObject()
                    .field("bookid", bookid)
                    .field("bookprice", bookprice)
                    .endObject();
            IndexResponse indexResponse = client.prepareIndex("tao","book").setSource(xContentBuilder).get();
            return "新增图书信息完成,图书信息ID为:"+indexResponse.getId();

        }catch (IOException ex){
            ex.printStackTrace();
            return "新增图书信息失败,失败信息:"+ex.getMessage();
        }


    }

实现删除数据

需求:根据给定的id,来删除数据信息

    /**
     * 从ES中删除一条数据
     * @param bookid
     * @return
     */
    @RequestMapping(value = "/del",method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
    @ResponseBody
    public String delInfo(String bookid){
        if (Strings.isNullOrEmpty(bookid)){
            return "参数不存在或者参数异常";
        }
        DeleteResponse deleteResponse = client.prepareDelete("tao", "book", bookid).get();
        //将会返回两个结果:NOT_FOUND 未发现 DELETED 已删除
        return deleteResponse.getResult().toString() ;
    }

实现修改数据

修改指定id的数据信息

    /**
     * 根据Id简单的修改信息
     * @param bookid
     * @param bookprice
     * @return
     */
    @RequestMapping(value = "/modify",method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
    @ResponseBody
    public String nodify(String bookid,@RequestParam(required = false) Float bookprice){

        UpdateRequest request = new UpdateRequest("tao","book",bookid);
        try {
            XContentBuilder xContentBuilder = XContentFactory.jsonBuilder().startObject();
            if (bookprice.floatValue() <= 0){
                return "参数异常或者参数格式错误!";
            }else {
                xContentBuilder.field("bookprice",bookprice).endObject();
            }
            request.doc(xContentBuilder);
            UpdateResponse updateResponse = client.update(request).get();
            return updateResponse.getResult().toString();

        }catch (Exception e) {
            e.printStackTrace();
            return "更新数据出现异常,异常信息:"+e.getMessage();
        }
    }

实现复合查询数据

需求:精准查询bookid,范围查询bookprice,gt_bookprice 最小的价格,lt_bookprice 最大价格

    /**
     * 使用ES进行复杂查询
     * @param bookid
     * @param gt_bookprice
     * @param lt_bookprice
     * @return
     */
    @RequestMapping(value = "/getConditon",method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_UTF8_VALUE)
    @ResponseBody
    public String querty(@RequestParam(required = false) String bookid,@RequestParam(defaultValue = "0.0") Float gt_bookprice,@RequestParam(required = false)Float lt_bookprice){
        //构造查询条件
        BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();
        if (!Strings.isNullOrEmpty(bookid)){
            boolQueryBuilder.must(QueryBuilders.matchQuery( "bookid",bookid));
        }

        //构造范围查询
        RangeQueryBuilder rangeQuery = QueryBuilders.rangeQuery("bookprice").from(gt_bookprice,true);
         if(lt_bookprice!=null && lt_bookprice.floatValue() > 0){
             rangeQuery.to(lt_bookprice);
         }
         //整合为一个查询条件
         boolQueryBuilder.filter(rangeQuery);

        SearchRequestBuilder searchRequestBuilder = client.prepareSearch("tao")
                .setTypes("book")
                .setSearchType(SearchType.DFS_QUERY_THEN_FETCH)
                .setQuery(boolQueryBuilder)
                .setFrom(0)
                .setSize(10);
        //可以直接输出,方便查看请求体
        System.out.println("查询条件:"+searchRequestBuilder);
        SearchResponse searchResponse = searchRequestBuilder.get();
        List<Map> result = new ArrayList<>();
        //从查询接口中遍历数据进行封装操作
        for (SearchHit hitFields:searchResponse.getHits()){
            result.add(hitFields.getSource());
        }
        return result.toString();
    }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,650评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,802评论 6 342
  • 第五章 回到表面、设计顾客体验 执行就是一切。必须要像对待其他产品那样,精心打造并精确设计用户体验。 设计思维不仅...
    灵儿Amelie阅读 506评论 0 0
  • 有时候我们说有时间见,可能永远也没时间,我们说再见却是再也不见! 好像在最好的年纪变成了最好的自己,却貌似错...
    丑妞阅读 291评论 0 0
  • 流光容易把人抛, 红了樱桃,绿了芭蕉。 时光里,我们的故事一章又一章。 岁月流逝,老却了容颜,成熟了心智。 年少时...
    牛凌阅读 680评论 1 2