Elasticsearch是一个基于Lucene的搜索服务器。它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口。Elasticsearch是用Java语言开发的,并作为Apache许可条款下的开放源码发布,是一种流行的企业级搜索引擎。Elasticsearch用于云计算中,能够达到实时搜索,稳定,可靠,快速,安装使用方便。官方客户端在Java、.NET(C#)、PHP、Python、Apache Groovy、Ruby和许多其他语言中都是可用的。根据DB-Engines的排名显示,Elasticsearch是最受欢迎的企业搜索引擎,其次是Apache Solr,也是基于Lucene
演示:京东,淘宝
重要特性:
倒排索引的概念是基于MySQL这样的正向索引而言的。
如果用id索引一个表,如果是根据id查询,那么直接走索引,查询速度非常快。
但如果是基于表做模糊查询,只能是逐行扫描数据,流程如下:
先扫描id为一的,要一个一个获取数据,要进行全表查询,随着数据量增加,其查询效率也会越来越低。当数据量达到数百万时,就是一场灾难。
倒排索引中有两个非常重要的概念:
创建倒排索引是对正向索引的一种特殊处理,流程如下:
id | title | price |
---|---|---|
1 | 小米手机 | 3999 |
2 | 华为手机 | 4999 |
3 | 华为小米充电器 | 89 |
4 | 小米手环 | 299 |
正向索引 |
词条(term) | 文档id |
---|---|
小米 | 1,3,4 |
手机 | 1,2 |
华为 | 2,3 |
充电器 | 3 |
手环 | 4 |
倒排索引 |
倒排索引的搜索流程如下(以搜索"华为手机"为例):
那么为什么一个叫做正向索引,一个叫做倒排索引呢?
正向索引:
倒排索引:
elasticsearch中有很多独有的概念,与mysql中略有差别,但也有相似之处。
elasticsearch是面向文档(Document)存储的,可以是数据库中的一条商品数据,一个订单信息。文档数据会被序列化为json格式后存储在elasticsearch中:
而Json文档中往往包含很多的字段(Field),类似于数据库中的列。
索引(index),就是相同类型的文档的集合
例如:
![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/f5b81a4485134745a463a497e8af8d88.png#pic_center
因此,我们可以把索引当做是数据库中的表
数据库的表会有约束信息,用来定义表的结构、字段的名称、类型等信息。因此,索引库中就有映射(mapping),是索引中文档的字段约束信息,类似表的结构约束
我们统一的把mysql与elasticsearch的概念做一下对比:
MySQL | Elasticsearch | 说明 |
---|---|---|
Table | Index | 索引(index),就是文档的集合,类似数据库的表(table) |
Row | Document | 文档(Document),就是一条条的数据,类似数据库中的行(Row),文档都是JSON格式 |
Column | Field | 字段(Field),就是JSON文档中的字段,类似数据库中的列(Column) |
Schema | Mapping | Mapping(映射)是索引中文档的约束,例如字段类型约束。类似数据库的表结构(Schema) |
SQL | DSL | DSL是elasticsearch提供的JSON风格的请求语句,用来操作elasticsearch,实现CRUD |
因此在企业中,往往是两者结合使用:
分词器的作用是什么?
IK分词器有几种模式?
索引库就类似数据库表,mapping映射就类似表的结构。
我们要向es中存储数据,必须先创建“库”和“表”。
mapping是对索引库中文档的约束,常见的mapping属性包括:
例如下面的json文档:
{
"age": 18,
"weight": 70.2,
"isMarried": false,
"info": "apesourceJavaEE王讲师",
"email": "[email protected]",
"score": [99.1, 99.5, 98.9],
"name": {
"firstName": "师傅",
"lastName": "王"
}
}
对应的每个字段映射(mapping):
基本语法:
示例:
{
"mappings": {
"properties":{
"age": {
"type":"integer"
},
"weight": {
"type":"float"
},
"isMarried": {
"type":"boolean"
},
"info": {
"type":"text",
"analyzer":"ik_smart"
},
"email": {
"type":"keyword",
"index":"false"
},
"name": {
"properties": {
"firstName": {
"type":"keyword"
}
}
}
}
}
}
基本语法:
格式:
GET /索引库名
倒排索引结构虽然不复杂,但是一旦数据结构改变(比如改变了分词器),就需要重新创建倒排索引,这简直是灾难。因此索引库一旦创建,无法修改mapping。
虽然无法修改mapping中已有的字段,但是却允许添加新的字段到mapping中,因为不会对倒排索引产生影响。
语法说明:
PUT /索引库名/_mapping
{
"properties": {
"新字段名":{
"type": "integer"
}
}
}
语法:
格式:
DELETE /索引库名
总结:
语法:
POST /索引库名/_doc/文档id
{
"字段1": "值1",
"字段2": "值2",
"字段3": {
"子属性1": "值3",
"子属性2": "值4"
},
// ...
}
示例:
POST /apesource/_doc/1
{
"info": "apesource程序员Java讲师",
"email": "[email protected]",
"name": {
"firstName": "师傅",
"lastName": "王"
}
}
根据rest风格,新增是post,查询应该是get,不过查询一般都需要条件,这里我们把文档id带上。
语法:
GET /{索引库名称}/_doc/{id}
删除使用DELETE请求,同样,需要根据id进行删除:
语法:
DELETE /{索引库名}/_doc/id值
修改有两种方式:
注意:如果根据id删除时,id不存在,第二步的新增也会执行,也就从修改变成了新增操作了。
语法:
PUT /{索引库名}/_doc/文档id
{
"字段1": "值1",
"字段2": "值2",
// ... 略
}
示例:
PUT /apesource/_doc/1
{
"info": "java程序员高级Java讲师",
"email": "[email protected]",
"name": {
"firstName": "师傅",
"lastName": "王"
}
}
增量修改:
增量修改是只修改指定id匹配的文档中的部分字段。
语法:
POST /{索引库名}/_update/文档id
{
"doc": {
"字段名": "新的值",
}
}
示例:
POST /apesource/_update/1
{
"doc": {
"email": "[email protected]"
}
}
总结:
**增量修改:**
增量修改是只修改指定id匹配的文档中的部分字段。
**语法:**
```json
POST /{索引库名}/_update/文档id
{
"doc": {
"字段名": "新的值",
}
}
示例:
POST /apesource/_update/1
{
"doc": {
"email": "[email protected]"
}
}
总结: