Elasticsearch8入门

版本

8.4

安装

windows本地安装,直接下载zip包解压即可
windows安装官方教程
启动

.\bin\elasticsearch.bat

elasticsearch本地就跑起来了

docker安装更方便

配置

主配置文件在config下的elasticsearch.yml

elasticsearch默认是要校验身份的,可以暂时去掉xpack.security.enabled修改为false

# Enable security features
xpack.security.enabled: false

配置可跨域(否则跨域不可访问)

http.cors.enabled: true
http.cors.allow-origin: "*"
http.host: 0.0.0.0
elasticsearch.yml

重启之后既可以开始使用了,简单测试一下是否正常

$ curl localhost:9200
localhost:9200

elasticsearch本地就搭建好了

客户端

就像mysql有navicat等客户端,用起来方便又直观,elasticsearch也有一个勉强可用的客户端
elasticsearch-head,是一个前端的代码,下载下来,运行也很简单: npm install安装依赖,npm run start运行,打开界面即可http://localhost:9100/

http://localhost:9100

但从实际效果上看,客户端提供的功能非常少,也不适合初学者学习

概念

Elasticsearch可以看做一个便于搜索的数据库,相比传统关系型数据库如下

Mysql ‐> Databases ‐> Tables ‐> Rows ‐> Columns
Elasticsearch ‐> Indices ‐> Types ‐> Documents ‐> Fields

Elasticsearch 8 已经删除了Types,所以现在的层级应该是

Elasticsearch ‐> Indices ‐> Documents ‐> Fields

Index

索引,相当于mysql中的一个数据表(原来有type时它相当于数据库,type相当于数据表,现在es8取消了type,index当做一个数据表来理解方便一点)

Documents

一条存储的数据,相当于mysql的一条数据

Fields

字段,类似mysql的子墩

API

Elasticsearch牛在封装了Lucene,提供了强大的搜索功能,同时对外暴露Restful Api使其易于对接,接下来就通过调用api来实现一个小功能:

存储一个网站的文章,并可以通过标题和内容智能检索

第一步,创建文章索引

就好比在mysql中创建一个文章表article,包含三个字段id,title,content

curl --location --request PUT 'http://127.0.0.1:9200/article' \
--header 'Content-Type: application/json' \
--data-raw '{
    "mappings": {
        "properties": {
            "id": {
                "type": "long",
                "index": false
            },
            "title": {
                "type": "text",
                "analyzer": "standard"
            },
            "content": {
                "type": "text",
                "analyzer": "standard"
            }
        }
    }
}'

关于索引的文档,可以参考Index APIs

第二步,添加几个测试文章

curl --location --request POST 'http://127.0.0.1:9200/article/_doc/1' \
--header 'Content-Type: application/json' \
--data-raw '{
    "id": 1,
    "title": "小李飞刀",
    "content": "小李飞刀,例无虚发"
}'

其中url上article代表索引名,_doc是默认,1代表id,如果没有则自动生成,请求体即文章的标题和内容

多增加几个文章进行测试

article

关于文档的文档,可以参考Document APIs

第三步,检索

测试数据准备好了,接下来进行搜索测试,我们搜索"刀",并希望所有带"刀"字样的都搜出来,相当于sql的

WHERE  (title like '%刀%' OR content like '%刀%')

写法如下,使用布尔过滤器

curl --location --request GET 'http://localhost:9200/article/_search' \
--header 'Content-Type: application/json' \
--data-raw '{
  "query": { 
    "bool": { 
      "should": [
        { "match": { "title":   "刀" }},
        { "match": { "content": "刀" }}
      ]
    }
  }
}'

最终输出结果如下

可以看到实现了我们的需求,带“刀”的数据都被查了出来
再搜索一下"刀剑"

刀剑

可以看到相关的都被查出,这种查询在mysql是做不到的

这种搜索查询的语法是ES自己指定的,可以支持很多复杂的查询方式,而且有一个很不错的中文文档,文档可能有些过期,但大部分还好,并且人性化的对照了mysql的场景

整合springboot

  • 如果想简单粗暴的代码调用es,其实使用HttpClient也可以,毕竟都是http接口,道理很简单,但写的代码肯定很费劲

  • 为了方便用户使用,es肯定是要提供sdk的,即elasticsearch-rest-high-level-client,专门针对es的客户端,使用起来肯定更方便

  • 为了使用更加方便,spring在elasticsearch-client基础上再次封装出了spring-data-elasticsearch,进一步的封装使spring用户调用es非常方便

spring每个阶段会根据spring的当前版本以及es的版本在不同时期适配不同的依赖包,这就导致每个spring-data-elasticsearch版本肯定会有一个适用的spring版本及es版本,当前的关系如下

spring-data-elasticsearch

很不幸,当前好像并没有适配es8的版本,且我自己用的springboot版本为2.3.x,所以只能尝试使用4.0.x的spring-data-elasticsearch,考虑到这种牛逼项目都有低版本适配,问题应该也不大

引入依赖
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-elasticsearch</artifactId>
</dependency>
配置Elasticsearch

主要是配置服务器地址,超时时间什么的,创建一个RestHighLevelClient客户端的bean

@Configuration
public class ElasticsearchClientConfig extends AbstractElasticsearchConfiguration {
   
    // 地址
    @Value("${spring.data.elasticsearch.host}")
    private String host;
    // 端口
    @Value("${spring.data.elasticsearch.port}")
    private String port;

    @Override
    @Bean
    public RestHighLevelClient elasticsearchClient() {
        final ClientConfiguration clientConfiguration = ClientConfiguration.builder()
                .connectedTo(host + ":" + port)
                .build();
        return RestClients.create(clientConfiguration).rest();
    }
}

到此spring的es配置结束,接下来就是增删查改,用过jpa的会发现基本一个套路

定义实体,映射到索引

创建一个实体Book并映射到es的索引book

@Data
@Document(indexName = "book",createIndex = true) // 映射book索引
public class Book {
     // 主键标识
    @Id
    @Field(type = FieldType.Long)
    private Long id;
    @Field(analyzer="standard")
    private String title;
    @Field(analyzer="standard")
    private String content;
}
定义仓库
public interface BookRepositoryImpl extends ElasticsearchRepository<Book, Long> {
    List<Book> findByTitleOrContent(String title, String content);
}

这个仓库的接口定义好之后,就可以通过spring注入,注入对象默认包含方法

  • save 保存,新增/修改
  • findAll 查询,带分页
  • delete 删除
  • 等等

也可以自己定义方法,主要有两种方式
一、直接从方法名称派生查询
例如上面的findByTitleOrContent就会自动转化为根据标题或内容查询
二、使用@Query注解自定义的查询
例如

interface BookRepository extends ElasticsearchRepository<Book, String> {
    @Query("{\"match\": {\"name\": {\"query\": \"?0\"}}}")
    Page<Book> findByName(String name,Pageable pageable);
}

使用过jpa应该再熟悉不过了,功能十分强大,可以参考官方文档

使用仓库

这就非常简单了,写个小例子

@Autowired
private BookRepository bookRepository;

public void test() {
    Book book = new Book();
    book.setId(1L);
    book.setTitle("雪中悍刀行");
    book.setContent("你是我的白月光,我是你的小砒霜");
    // 新增文档
    bookRepository.save(book);
    // 查询所有
    Iterable<Book> all = bookRepository.findAll();
    for (book po : all) {

    }
    // 通过标题或内容查询带"刀"的文档
    List<Book> hits = bookRepository.findByTitleOrContent("刀", "刀");
}
修改版本不兼容问题

跑了一下还出出问题,报错type不能为null,原因是es8已取消type,而我们低版本客户端还是会校验type必须存在,导致运行出错,报错点如下


type不能为null

修改办法:
没找到太好的解决办法,只是粗暴的重写了DocWriteResponse类


type不能为null解决
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容