搜索引擎Elasticsearch

1、概述

ES介绍

  • The Elastic Stack,包括ElasticsearchKibanaBeatsLogstash(也称为ELK Stack)。 能够安全可靠地获取任何来源、任何格式的数据,然后实时地对数据进行搜索、分析和可视化。Elaticsearch,简称为ES,**ES是一个开源的高扩展的分布式全文搜索引擎**,是整个Elastic Stack技术栈的核心。它可以近乎实时的存储、检索数据;本身扩展性很好,可以扩展到上百台服务器,处理PB级别的数据。

    全文搜索引擎

  • Google,百度类的网站搜索,它们都是根据网页中的关键字生成索引,我们在搜索的时候输入关键字,它们会将该关键字即索引匹配到的所有网页返回;还有常见的项目中应用日 志的搜索等等。对于这些非结构化的数据文本,关系型数据库搜索不是能很好的支持。

  • 一般传统数据库,全文检索都实现的很鸡肋,因为一般也没人用数据库存文本字段。进 行全文检索需要扫描整个表,如果数据量大的话即使对SQL的语法优化,也收效甚微。建立了索引,但是维护起来也很麻烦,对于insertupdate操作都会重新构建索引。

  • 基于以上原因可以分析得出,在一些生产环境中,使用常规的搜索方式,性能是非常差的:

    • 搜索的数据对象是大量的非结构化的文本数据。
    • 文件记录量达到数十万或数百万个甚至更多。
    • 支持大量基于交互式文本的查询。
    • 需求非常灵活的全文搜索查询。
    • 对高度相关的搜索结果的有特殊需求,但是没有可用的关系数据库可以满足。
    • 对不同记录类型、非文本数据操作或安全事务处理的需求相对较少的情况。为了解决结构化数据搜索和非结构化数据搜索性能问题,我们就需要专业,健壮,强大的全文搜索引擎。
  • 这里说到的全文搜索引擎指的是目前广泛应用的主流搜索引擎。它的工作原理是计算机索引程序通过扫描文章中的每一个词,对每一个词建立一个索引,指明该词在文章中出现的次数和位置,当用户查询时,检索程序就根据事先建立的索引进行查找,并将查找的结果反馈给用户的检索方式。这个过程类似于通过字典中的检索字表查字的过程。

2、ES安装

  • 以下是在centos7服务器上使用docker安装ES7.8.0单机版本的步骤。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    # 拉取镜像
    docker pull elasticsearch:7.8.0

    # 创建相应配置文件目录
    mkdir -p /mydata/elasticsearch/{config,data,plugins}
    echo "http.host: 0.0.0.0" > /mydata/elasticsearch/config/elasticsearch.yml

    # 赋予权限
    chmod -R 775 /mydata/elasticsearch/

    # 运行elasticsearch:7.8.0
    docker run --name elasticsearch -p 9200:9200 -p 9300:9300 \
    -e "discovery.type=single-node" \
    -e ES_JAVA_OPTS="-Xms64m -Xmx128m" \
    -v /mydata/elasticsearch/config/elasticsearch.yml:/usr/share/elasticsearch/config/elasticsearch.yml \
    -v /mydata/elasticsearch/data:/usr/share/elasticsearch/data \
    -v /mydata/elasticsearch/plugins:/usr/share/elasticsearch/plugins \
    -d elasticsearch:7.8.0
  • 安装完毕后,可查询到相关信息。

    1
    2
    # 查看集群健康状态
    curl 127.0.0.1:9200/_cat/health?v
    1
    2
    # 查看节点健康状态
    curl 127.0.0.1:9200/_cat/nodes?v
    1
    2
    # 查看索引信息
    curl 127.0.0.1:9200/_cat/indices?v
    1
    2
    # 查看版本
    curl http://127.0.0.1:9200/

3、ES基础知识

3.1 数据格式

  • Elasticsearch是面向文档型数据库,一条数据在这里就是一个文档。Elasticsearch里存储文档数据和关系型数据库MySQL存储数据的概念进行一个类比。

  • ES里的Index可以看做一个库,而Types相当于表,Documents则相当于表的行。这里Types的概念已经被逐渐弱化,Elasticsearch 6.X中,一个index下已经只能包含一个typeElasticsearch 7.X中,Type的概念已经被删除了。

3.2 索引操作

3.2.1 创建索引

  • 对比关系型数据库,创建索引就等同于创建数据库,在Postman中,向ES服务器发PUT请求。

    1
    2
    # 发送PUT请求创建一个名为shopping的索引
    http://127.0.0.1:9200/shopping
  • 请求后响应结果如下:

    1
    2
    3
    4
    5
    6
    {
    "acknowledged": true, // 响应结果
    "shards_acknowledged": true, // 分片结果
    "index": "shopping" // 索引名称
    }
    // 注意:创建索引库的分片数默认1片,在7.0.0之前的Elasticsearch版本中,默认5片
  • 如果重复添加索引,会返回错误信息。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    {
    "error": {
    "root_cause": [
    {
    "type": "resource_already_exists_exception",
    "reason": "index [shopping/9pfdCtBKSGG4TVXnoPvs0Q] already exists",
    "index_uuid": "9pfdCtBKSGG4TVXnoPvs0Q",
    "index": "shopping"
    }
    ],
    "type": "resource_already_exists_exception",
    "reason": "index [shopping/9pfdCtBKSGG4TVXnoPvs0Q] already exists",
    "index_uuid": "9pfdCtBKSGG4TVXnoPvs0Q",
    "index": "shopping"
    },
    "status": 400
    }

3.2.2 查看索引

  • 查看指定索引。

    1
    2
    # 发送GET请求查看名为shopping的索引
    http://127.0.0.1:9200/shopping
    • 请求后响应结果如下:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      {
      "shopping": { // 索引名
      "aliases": {}, // 别名
      "mappings": {}, // 映射
      "settings": { // 设置
      "index": { // 索引
      "creation_date": "1645104162919", // 创建时间
      "number_of_shards": "1", // 主分片数量
      "number_of_replicas": "1", // 副分片数量
      "uuid": "9pfdCtBKSGG4TVXnoPvs0Q", // 唯一标识
      "version": { // 版本
      "created": "7080099"
      },
      "provided_name": "shopping" // 名称
      }
      }
      }
      }
  • 查看所有索引。

    1
    2
    # 发送GET请求
    http://127.0.0.1:9200/_cat/indices?v
    • 这里请求路径中的_cat表示查看的意思,indices表示索引,所以整体含义就是查看当前ES服务器中的所有索引,就好像MySQL中的show tables的感觉,服务器响应结果如下:

      1
      2
      health status index    uuid                   pri rep docs.count docs.deleted store.size pri.store.size
      yellow open shopping 9pfdCtBKSGG4TVXnoPvs0Q 1 1 0 0 208b 208b

3.2.3 删除索引

  • 发送DELETE请求。

    1
    2
    # 发送DELETE请求删除名为shopping的索引
    http://127.0.0.1:9200/shopping
  • 返回结果如下。

    1645106139424
  • 重新访问索引时,服务器返回响应:索引不存在。

    1645106186851

3.3 文档操作

3.3.1 创建文档

  • 索引已经创建好了,接下来创建文档,并添加数据。这里的文档可以类比为关系型数据库中的表数据,添加的数据格式为JSON格式。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    # 发送POST请求
    http://127.0.0.1:9200/shopping/_doc # 或者http://127.0.0.1:9200/shopping/_create

    # 请求体如下
    {
    "title":"小米手机",
    "category":"小米",
    "price":3999.00
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    {
    "_index": "shopping", // 索引
    "_type": "_doc", // 类型-文档
    "_id": "OR3mCn8BLA16qSV9gGDY", // 唯一标识,可以类比为MySQL中的主键,随机生成
    "_version": 1, // 版本
    "result": "created", // 结果,这里的create表示创建成功
    "_shards": { // 分片
    "total": 2, // 总数
    "successful": 1, // 成功
    "failed": 0 // 失败
    },
    "_seq_no": 0,
    "_primary_term": 1
    }
    1645155498365
  • 上面的数据创建后,由于没有指定数据唯一性标识(ID),默认情况下,ES服务器会随机生成一个。如果想要自定义唯一性标识,需要在创建时指定:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    # 发送POST请求
    # 此处需要注意:如果增加数据时明确数据主键,那么请求方式也可以为PUT
    http://127.0.0.1:9200/shopping/_doc/1

    # 请求体如下
    {
    "title":"华为手机",
    "category":"华为u",
    "price":3999.00
    }
    1645156331586

3.3.2 查看文档

  • 查看文档时,需要指明文档的唯一性标识,类似于MySQL中数据的主键查询。

    1
    2
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_doc/1
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    {
    "_index": "shopping", // 索引
    "_type": "_doc", // 文档类型
    "_id": "1",
    "_version": 1,
    "_seq_no": 1,
    "_primary_term": 1,
    "found": true, // 查询结果,true表示查找到,false表示未查找到
    "_source": { // 文档源信息
    "title": "华为手机",
    "category": "华为u",
    "price": 3999.00
    }
    }
    1645156844110
  • 查看全部数据:

    1
    2
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_search
    1645157422241
  • 如果查询不到数据则:

    1645157239550

3.3.3 修改文档

全量修改

  • 和新增文档一样,输入相同的URL地址请求,如果请求体变化,会将原有的数据内容覆盖。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    # 发送POST请求
    http://127.0.0.1:9200/shopping/_doc/1

    # 请求体如下
    {
    "title":"小米手机2",
    "category":"小米2",
    "price":3999.00
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_version": 2, // 版本
    "result": "updated", // updated表示数据被更新
    "_shards": {
    "total": 2,
    "successful": 1,
    "failed": 0
    },
    "_seq_no": 4,
    "_primary_term": 1
    }
    1645158397099

局部修改

  • 修改数据时,也可以只修改某一条数据的局部信息。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    # 发送POST请求
    http://127.0.0.1:9200/shopping/_update/1

    # 请求体如下
    {
    "doc": {
    "title":"小米手机",
    "category":"小米"
    }
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_version": 4,
    "result": "updated",
    "_shards": {
    "total": 2,
    "successful": 1,
    "failed": 0
    },
    "_seq_no": 6,
    "_primary_term": 1
    }
    1645158234120
  • 修改完成后查看数据:

    1645158477319

3.3.4 删除文档

  • 删除一个文档不会立即从磁盘上移除,它只是被标记成已删除(逻辑删除)。

    1
    2
    # 发送DELETE请求
    http://127.0.0.1:9200/shopping/_doc/1
  • 返回结果:

    1645158653368
  • 删除后再查询当前文档信息:

    1645158734516
  • 如果删除一个并不存在的文档会返回not_found表示未查找到:

    1645158773375

3.3.5 条件删除文档

  • 一般删除数据都是根据文档的唯一性标识进行删除,实际操作时,也可以根据条件对多条数据进行删除。

  • 首先增加多条数据:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    # 发送POST请求
    http://127.0.0.1:9200/shopping/_doc/1

    # 请求体如下
    {
    "title":"小米手机",
    "category":"小米",
    "price":3999.00
    }
    1645159360374
    1
    2
    3
    4
    5
    6
    7
    8
    9
    # 发送POST请求
    http://127.0.0.1:9200/shopping/_doc/2

    # 请求体如下
    {
    "title":"华为手机",
    "category":"华为",
    "price":3999.00
    }
    1645159396672
  • 发送条件删除请求:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送POST请求
    http://127.0.0.1:9200/shopping/_delete_by_query

    # 请求体如下
    {
    "query":{
    "match":{
    "price": 3999.00
    }
    }
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    {
    "took": 266, // 耗时
    "timed_out": false, // 是否超时
    "total": 2, // 总数
    "deleted": 2, // 删除数量
    "batches": 1,
    "version_conflicts": 0,
    "noops": 0,
    "retries": {
    "bulk": 0,
    "search": 0
    },
    "throttled_millis": 0,
    "requests_per_second": -1.0,
    "throttled_until_millis": 0,
    "failures": []
    }
    1645165928097

3.3.6 条件查询文档

URL带参查询

1
2
# 发送GET请求,查找category为小米的文档
http://127.0.0.1:9200/shopping/_search?q=category:小米
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    {
    "took": 4,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 1,
    "relation": "eq"
    },
    "max_score": 1.3862942,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.3862942,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    }
    ]
    }
    }
1645167015543
  • 上述为URL带参数形式查询,这很容易让不善者心怀恶意,或者参数值出现中文会出现乱码情况。为了避免这些情况,我们可用使用带JSON请求体请求进行查询。

请求体带参查询

1
2
3
4
5
6
7
8
9
10
11
# 发送GET请求,查找category为小米的文档
http://127.0.0.1:9200/shopping/_search

# 请求体如下
{
"query":{
"match":{
"category":"小米"
}
}
}
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    {
    "took": 2,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 1,
    "relation": "eq"
    },
    "max_score": 1.3862942,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.3862942,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    }
    ]
    }
    }
1645167308424

请求体带参全量查询

1
2
3
4
5
6
7
8
9
# 发送GET请求,查找shopping索引下所有文档的内容
http://127.0.0.1:9200/shopping/_search

# 请求体如下
{
"query":{
"match_all":{}
}
}
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    {
    "took": 1,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": 1.0,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.0,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 1.0,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    }
    }
    ]
    }
    }
1645168091599

3.3.7 分页查询

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
# 发送GET请求,分页查询从0索引开始的2条数据。如果要查第2页,每页数量为2,则form为2((页码-1)*每页数量),size为2
http://127.0.0.1:9200/shopping/_search

# 请求体如下
{
"query":{
"match_all":{}
},
"from":0,
"size":2
}
# 如果只想查询title字段,则:
{
"query":{
"match_all":{}
},
"from":0,
"size":2,
"_source": ["title"]
}
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    {
    "took": 6,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": 1.0,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.0,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 1.0,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    }
    }
    ]
    }
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    {
    "took": 2,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": 1.0,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.0,
    "_source": {
    "title": "小米手机"
    }
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 1.0,
    "_source": {
    "title": "华为手机"
    }
    }
    ]
    }
    }
1645168963412 1645169218543

3.3.8 排序

1
2
3
4
5
6
7
8
9
10
11
12
13
14
# 发送GET请求,按照价格降序排序
http://127.0.0.1:9200/shopping/_search

# 请求体如下
{
"query":{
"match_all":{}
},
"sort":{
"price":{
"order":"desc"
}
}
}
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    {
    "took": 3,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": null,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": null,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    },
    "sort": [
    3999.0
    ]
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": null,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    },
    "sort": [
    3999.0
    ]
    }
    ]
    }
    }
1645169554683

3.3.9 多条件查询

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
# 发送GET请求
http://127.0.0.1:9200/shopping/_search

# 请求体如下,查询小米牌子,价格为3999元的数据,must相当于数据库的and
{
"query":{
"bool":{
"must":[{
"match":{
"category":"小米"
}
},{
"match":{
"price":3999.00
}
}]
}
}
}

# 请求体如下,查询小米或华为牌子的数据,should相当于数据库的or
{
"query":{
"bool":{
"should":[{
"match":{
"category":"小米"
}
},{
"match":{
"category":"华为"
}
}]
}
}
}
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    {
    "took": 3,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 1,
    "relation": "eq"
    },
    "max_score": 2.3862944,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 2.3862944,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    }
    ]
    }
    }
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    {
    "took": 3,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": 1.3862942,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.3862942,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 1.3862942,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    }
    }
    ]
    }
    }
1645169972050 1645170210114

3.3.10 范围查询

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
# 发送GET请求
http://127.0.0.1:9200/shopping/_search

# 请求体如下,查询小米和华为的牌子,价格大于2000元的手机
{
"query":{
"bool":{
"should":[{
"match":{
"category":"小米"
}
},{
"match":{
"category":"华为"
}
}],
"filter":{
"range":{
"price":{
"gt":2000
}
}
}
}
}
}
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    {
    "took": 3,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": 1.3862942,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 1.3862942,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 1.3862942,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    }
    }
    ]
    }
    }
1645170603857

3.3.11 全文检索

  • match匹配类型查询,会把查询条件进行分词,然后进行查询,多个词条之间是or的关系。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_search

    # 请求体如下,返回结果带回品牌有“小”或者“华”的数据
    {
    "query":{
    "match":{
    "category" : "小华"
    }
    }
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    {
    "took": 2,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 2,
    "relation": "eq"
    },
    "max_score": 0.6931471,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "1",
    "_score": 0.6931471,
    "_source": {
    "title": "小米手机",
    "category": "小米",
    "price": 3999.00
    }
    },
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 0.6931471,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    }
    }
    ]
    }
    }

3.3.12 完全匹配

  • 可以使用match_phrase进行完全匹配,它是一个整体的模糊查询。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_search

    # 请求体如下,返回结果带回品牌有“华为”的数据
    {
    "query":{
    "match_phrase":{
    "category" : "华为"
    }
    }
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    {
    "took": 1,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 1,
    "relation": "eq"
    },
    "max_score": 0.6931471,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 0.6931471,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    }
    }
    ]
    }
    }

3.3.13 高亮查询

  • 在进行关键字搜索时,搜索出的内容中的关键字会显示不同的颜色,称之为高亮。

  • Elasticsearch可以对查询内容中的关键字部分,进行标签和样式(高亮)的设置。

  • 在使用match查询的同时,加上一个highlight属性:

    • pre_tags:前置标签。
    • post_tags:后置标签。
    • fields:需要高亮的字段。
    • title:这里声明title字段需要高亮,后面可以为这个字段设置特有配置,也可以空。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_search

    # 请求体如下,返回结果带回品牌有“华为”的数据
    {
    "query": {
    "match_phrase": {
    "category" : "华为"
    }
    },
    "highlight": {
    "pre_tags": "<font color='red'>",
    "post_tags": "</font>",
    "fields": {
    "category": {}
    }
    }
    }
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    {
    "took": 9,
    "timed_out": false,
    "_shards": {
    "total": 1,
    "successful": 1,
    "skipped": 0,
    "failed": 0
    },
    "hits": {
    "total": {
    "value": 1,
    "relation": "eq"
    },
    "max_score": 1.3862942,
    "hits": [
    {
    "_index": "shopping",
    "_type": "_doc",
    "_id": "2",
    "_score": 1.3862942,
    "_source": {
    "title": "华为手机",
    "category": "华为",
    "price": 3999.00
    },
    "highlight": {
    "category": [
    "<font color='red'>华</font><font color='red'>为</font>"
    ]
    }
    }
    ]
    }
    }

3.3.14 聚合查询

  • 聚合允许使用者对es文档进行统计分析,类似与关系型数据库中的group by,当然还有很多其他的聚合,例如取最大值、平均值等等。

  • 对某个字段取最大值:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_search

    # 请求体如下,查询字段为price的最大值
    {
    "aggs": {
    "max_price": { # 随意起名
    "max": {
    "field": "price"
    }
    }
    },
    "size": 0
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      {
      "took": 2,
      "timed_out": false,
      "_shards": {
      "total": 1,
      "successful": 1,
      "skipped": 0,
      "failed": 0
      },
      "hits": {
      "total": {
      "value": 2,
      "relation": "eq"
      },
      "max_score": null,
      "hits": []
      },
      "aggregations": {
      "max_price": {
      "value": 3999.0
      }
      }
      }
  • 按某些字段进行分组:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    # 发送GET请求
    http://127.0.0.1:9200/shopping/_search

    # 请求体如下,按price进行分组
    {
    "aggs": {
    "price_group": { # 随意起名
    "terms": {
    "field": "price"
    }
    }
    },
    "size": 0
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      {
      "took": 7,
      "timed_out": false,
      "_shards": {
      "total": 1,
      "successful": 1,
      "skipped": 0,
      "failed": 0
      },
      "hits": {
      "total": {
      "value": 2,
      "relation": "eq"
      },
      "max_score": null,
      "hits": []
      },
      "aggregations": {
      "price_group": {
      "doc_count_error_upper_bound": 0,
      "sum_other_doc_count": 0,
      "buckets": [
      {
      "key": 3999.0,
      "doc_count": 2
      }
      ]
      }
      }
      }

3.4 映射操作

  • 有了索引库,等于有了数据库中的database
  • 接下来就需要建索引库(index)中的映射了,类似于数据库(database)中的表结构(table)。 创建数据库表需要设置字段名称,类型,长度,约束等;索引库也一样,需要知道这个类型下有哪些字段,每个字段有哪些约束信息,这就叫做映射(mapping)。
  • 映射数据说明:
    • 字段名:任意填写,下面指定许多属性,例如:title、subtitle、images、price
    • type:类型,Elasticsearch中支持的数据类型非常丰富,说几个关键的:
      • String类型,又分两种:
        • text:可分词。
        • keyword:不可分词,数据会作为完整字段进行匹配。
      • Numerical:数值类型,分两类。
        • 基本数据类型:long、integer、short、byte、double、float、half_float
        • 浮点数的高精度类型:scaled_float
      • Date:日期类型。
      • Array:数组类型。
      • Object:对象。
    • index:是否索引,默认为true,也就是说你不进行任何配置,所有字段都会被索引。
      • true:字段会被索引,则可以用来进行搜索。
      • false:字段不会被索引,不能用来搜索。
    • store:是否将数据进行独立存储,默认为false
      • 原始的文本会存储在_source里面,默认情况下其他提取出来的字段都不是独立存储的,是从_source里面提取出来的。当然你也可以独立的存储某个字段,只要设置"store":true即可,获取独立存储的字段要比从_source中解析快得多,但是也会占用更多的空间,所以要根据实际业务需求来设置。
    • analyzer:分词器,这里的ik_max_word即使用ik分词器。

3.4.1 创建映射

  • 先创建一个索引:

    1
    2
    # 发送PUT请求
    http://127.0.0.1:9200/user
    • 返回结果:

      1
      2
      3
      4
      5
      {
      "acknowledged": true,
      "shards_acknowledged": true,
      "index": "user"
      }
  • 创建映射。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    # 发送PUT请求
    http://127.0.0.1:9200/user/_mapping

    # 请求体如下
    {
    "properties": {
    "name":{
    "type": "text",
    "index": true
    },
    "sex":{
    "type": "keyword",
    "index": true
    },
    "tel":{
    "type": "keyword",
    "index": false
    }
    }
    }
    • 返回结果:

      1
      2
      3
      {
      "acknowledged": true
      }

3.4.2 查看映射

1
2
# 发送GET请求
http://127.0.0.1:9200/user/_mapping
  • 返回结果:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    {
    "user": {
    "mappings": {
    "properties": {
    "name": {
    "type": "text"
    },
    "sex": {
    "type": "keyword"
    },
    "tel": {
    "type": "keyword",
    "index": false
    }
    }
    }
    }
    }

3.4.3 测试映射

  • 往user索引添加数据。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    # 发送POST请求
    http://127.0.0.1:9200/user/_create/1001

    # 请求体如下
    {
    "name":"小明",
    "sex":"男的",
    "tel":"12345678987"
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      {
      "_index": "user",
      "_type": "_doc",
      "_id": "1001",
      "_version": 1,
      "result": "created",
      "_shards": {
      "total": 2,
      "successful": 1,
      "failed": 0
      },
      "_seq_no": 0,
      "_primary_term": 1
      }
  • 查找name含有”小”数据:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送GET请求
    http://127.0.0.1:9200/user/_search

    # 请求体如下
    {
    "query":{
    "match":{
    "name":"小"
    }
    }
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      {
      "took": 10,
      "timed_out": false,
      "_shards": {
      "total": 1,
      "successful": 1,
      "skipped": 0,
      "failed": 0
      },
      "hits": {
      "total": {
      "value": 1,
      "relation": "eq"
      },
      "max_score": 0.2876821,
      "hits": [
      {
      "_index": "user",
      "_type": "_doc",
      "_id": "1001",
      "_score": 0.2876821,
      "_source": {
      "name": "小明",
      "sex": "男的",
      "tel": "12345678987"
      }
      }
      ]
      }
      }
  • 查找sex含有”男”数据:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送GET请求
    http://127.0.0.1:9200/user/_search

    # 请求体如下
    {
    "query":{
    "match":{
    "sex":"男"
    }
    }
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      {
      "took": 1,
      "timed_out": false,
      "_shards": {
      "total": 1,
      "successful": 1,
      "skipped": 0,
      "failed": 0
      },
      "hits": {
      "total": {
      "value": 0,
      "relation": "eq"
      },
      "max_score": null,
      "hits": []
      }
      }
    • 找不到想要的结果,只因创建映射时”sex”的类型为”keyword”,需要完全匹配。

  • 查找sex含有”男的”数据:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送GET请求
    http://127.0.0.1:9200/user/_search

    # 请求体如下
    {
    "query":{
    "match":{
    "sex":"男的"
    }
    }
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      {
      "took": 1,
      "timed_out": false,
      "_shards": {
      "total": 1,
      "successful": 1,
      "skipped": 0,
      "failed": 0
      },
      "hits": {
      "total": {
      "value": 1,
      "relation": "eq"
      },
      "max_score": 0.2876821,
      "hits": [
      {
      "_index": "user",
      "_type": "_doc",
      "_id": "1001",
      "_score": 0.2876821,
      "_source": {
      "name": "小明",
      "sex": "男的",
      "tel": "12345678987"
      }
      }
      ]
      }
      }
  • 查询指定电话的数据:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    # 发送GET请求
    http://127.0.0.1:9200/user/_search

    # 请求体如下
    {
    "query":{
    "match":{
    "tel":"12345678987"
    }
    }
    }
    • 返回结果如下,因为创建映射时”tel”的”index”为false:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      {
      "error": {
      "root_cause": [
      {
      "type": "query_shard_exception",
      "reason": "failed to create query: Cannot search on field [tel] since it is not indexed.",
      "index_uuid": "W0wfSvpLSgexFhBf1J3WoQ",
      "index": "user"
      }
      ],
      "type": "search_phase_execution_exception",
      "reason": "all shards failed",
      "phase": "query",
      "grouped": true,
      "failed_shards": [
      {
      "shard": 0,
      "index": "user",
      "node": "cKIXbDc8SmWn4gjqUps8tw",
      "reason": {
      "type": "query_shard_exception",
      "reason": "failed to create query: Cannot search on field [tel] since it is not indexed.",
      "index_uuid": "W0wfSvpLSgexFhBf1J3WoQ",
      "index": "user",
      "caused_by": {
      "type": "illegal_argument_exception",
      "reason": "Cannot search on field [tel] since it is not indexed."
      }
      }
      }
      ]
      },
      "status": 400
      }

4、ES进阶

4.1 核心概念

4.1.1 索引(Index)

  • 一个索引就是一个拥有几分相似特征的文档的集合。比如说,你可以有一个客户数据的 索引,另一个产品目录的索引,还有一个订单数据的索引。一个索引由一个名字来标识(必须全部是小写字母),并且当我们要对这个索引中的文档进行索引、搜索、更新和删除的时 候,都要使用到这个名字。在一个集群中,可以定义任意多的索引。
  • 能搜索的数据必须索引,这样的好处是可以提高查询速度,比如:新华字典前面的目录就是索引的意思,目录可以提高查询速度。
  • Elasticsearch索引的精髓:一切设计都是为了提高搜索的性能。

4.1.2 类型(Type)

  • 在一个索引中,你可以定义一种或多种类型。

  • 一个类型是你的索引的一个逻辑上的分类/分区,其语义完全由你来定。通常,会为具 有一组共同字段的文档定义一个类型。不同的版本,类型发生了不同的变化。

4.1.3 文档(Document)

  • 一个文档是一个可被索引的基础信息单元,也就是一条数据。
  • 比如:你可以拥有某一个客户的文档,某一个产品的一个文档,当然,也可以拥有某个订单的一个文档。文档以JSON(Javascript Object Notation)格式来表示,而JSON是一个 到处存在的互联网数据交互格式。
  • 在一个index/type里面,你可以存储任意多的文档。

4.1.4 字段(Field)

  • 相当于是数据表的字段,对文档数据根据不同属性进行的分类标识。

4.1.5 映射(Mapping)

  • mapping是处理数据的方式和规则方面做一些限制,如:某个字段的数据类型、默认值、 分析器、是否被索引等等。这些都是映射里面可以设置的,其它就是处理ES里面数据的一 些使用规则设置也叫做映射,按着最优规则处理数据对性能提高很大,因此才需要建立映射, 并且需要思考如何建立映射才能对性能更好。

4.1.6 分片(Shards)

  • 一个索引可以存储超出单个节点硬件限制的大量数据。比如,一个具有10亿文档数据 的索引占据1TB的磁盘空间,而任一节点都可能没有这样大的磁盘空间。或者单个节点处理搜索请求,响应太慢。为了解决这个问题,Elasticsearch提供了将索引划分成多份的能力, 每一份就称之为分片。当你创建一个索引的时候,你可以指定你想要的分片的数量。每个分片本身也是一个功能完善并且独立的“索引”,这个“索引”可以被放置到集群中的任何节点上。
  • 分片很重要,主要有两方面的原因:
    • 允许你水平分割/扩展你的内容容量。
    • 允许你在分片之上进行分布式的、并行的操作,进而提高性能/吞吐量。
  • 至于一个分片怎样分布,它的文档怎样聚合和搜索请求,是完全由Elasticsearch管理的, 对于作为用户的你来说,这些都是透明的,无需过分关心。
  • 被混淆的概念是,一个Lucene索引我们在Elasticsearch称作分片 。 一个Elasticsearch索引是分片的集合。当Elasticsearch在索引中搜索的时候, 他发送查询到每一个属于索引的分片(Lucene索引),然后合并每个分片的结果到一个全局的结果集。

4.1.7 副本(Replicas)

  • 在一个网络/云的环境里,失败随时都可能发生,在某个分片/节点不知怎么的就处于离线状态,或者由于任何原因消失了,这种情况下,有一个故障转移机制是非常有用并且是 强烈推荐的。为此目的,Elasticsearch允许你创建分片的一份或多份拷贝,这些拷贝叫做复制分片(副本)。
  • 复制分片之所以重要,有两个主要原因:
    • 在分片/节点失败的情况下,提供了高可用性。因为这个原因,注意到复制分片从不与原/主要(original/primary)分片置于同一节点上是非常重要的。
    • 扩展你的搜索量/吞吐量,因为搜索可以在所有的副本上并行运行。
  • 总之,每个索引可以被分成多个分片。一个索引也可以被复制0次(意思是没有复制) 或多次。一旦复制了,每个索引就有了主分片(作为复制源的原来的分片)和复制分片(主分片的拷贝)之别。分片和复制的数量可以在索引创建的时候指定。在索引创建之后,你可 以在任何时候动态地改变复制的数量,但是你事后不能改变分片的数量。默认情况下,Elasticsearch中的每个索引被分片1个主分片和1个复制,这意味着,如果你的集群中至少有两个节点,你的索引将会有1个主分片和另外1个复制分片(1个完全拷贝),这样的话每个索引总共就有2个分片,我们需要根据索引需要确定分片个数。

4.1.8 分配(Allocation)

  • 将分片分配给某个节点的过程,包括分配主分片或者副本。如果是副本,还包含从主分片复制数据的过程。这个过程是由master节点完成的。

4.2 系统架构

1645241072665
  • 一个运行中的Elasticsearch实例称为一个节点,而集群是由一个或者多个拥有相同cluster.name配置的节点组成, 它们共同承担数据和负载的压力。当有节点加入集群中或者从集群中移除节点时,集群将会重新平均分布所有的数据。
  • 当一个节点被选举成为主节点时, 它将负责管理集群范围内的所有变更,例如增加、 删除索引,或者增加、删除节点等。 而主节点并不需要涉及到文档级别的变更和搜索等操作,所以当集群只拥有一个主节点的情况下,即使流量的增加它也不会成为瓶颈。 任何节点都可以成为主节点。我们的示例集群就只有一个节点,所以它同时也成为了主节点。
  • 作为用户,我们可以将请求发送到集群中的任何节点,包括主节点。每个节点都知道任意文档所处的位置,并且能够将我们的请求直接转发到存储我们所需文档的节点。 无论 我们将请求发送到哪个节点,它都能负责从各个包含我们所需文档的节点收集回数据,并将最终结果返回给客户端。Elasticsearch对这一切的管理都是透明的。

4.3 分布式集群

4.3.1 单节点集群

  • 我们在包含一个空节点的集群内创建名为users的索引,为了演示目的,我们将分配3个主分片和一份副本(每个主分片拥有一个副本分片)。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    # 发送PUT请求
    http://127.0.0.1:1001/users

    # 请求体如下
    {
    "settings" : {
    "number_of_shards" : 3,
    "number_of_replicas" : 1
    }
    }
  • 集群现在是拥有一个索引的单节点集群。所有3个主分片都被分配在node-1。

    1645242116341
  • 通过elasticsearch-head插件查看集群情况。

    • 当前我们的集群是正常运行的,但是在硬件故障时有丢失数据的风险。

4.3.2 故障转移

  • 当集群中只有一个节点在运行时,意味着会有一个单点故障问题——没有冗余。 幸运的是,我们只需再启动一个节点即可防止数据丢失。当你在同一台机器上启动了第二个节点时,只要它和第一个节点有同样的cluster.name配置,它就会自动发现集群并加入到其中。但是在不同机器上启动节点的时候,为了加入到同一集群,你需要配置一个可连接到的单播主机列表。之所以配置为使用单播发现,以防止节点无意中加入集群。只有在同一台机器上运行的节点才会自动组成集群。

  • 如果启动了第二个节点,我们的集群将会拥有两个节点的集群:所有主分片和副本分片都已被分配。

    1645242460728
  • 通过elasticsearch-head插件查看集群情况。

4.3.3 水平扩容

  • 怎样为我们的正在增长中的应用程序按需扩容呢?当启动了第三个节点,我们的集群将会拥有三个节点的集群:为了分散负载而对分片进行重新分配。

    1645242699396
  • 通过elasticsearch-head插件查看集群情况。

  • 但是如果我们想要扩容超过6个节点怎么办呢?

    • 主分片的数目在索引创建时就已经确定了下来。实际上,这个数目定义了这个索引能够存储的最大数据量。(实际大小取决于你的数据、硬件和使用场景) 。但是,读操作——搜索和返回数据——可以同时被主分片或副本分片所处理,所以当你拥有越多的副本分片时,也将拥有越高的吞吐量。

    • 在运行中的集群上是可以动态调整副本分片数目的,我们可以按需伸缩集群。让我们把副本数从默认的1增加到2。

      1
      2
      3
      4
      5
      6
      7
      # 发送PUT请求
      http://127.0.0.1:1001/users/_settings

      # 请求体如下
      {
      "number_of_replicas" : 2
      }
      • users索引现在拥有9个分片:3个主分片和6个副本分片。这意味着我们可以将集群扩容到9个节点,每个节点上一个分片。相比原来3个节点时,集群搜索性能可以提升3倍。

    • 通过elasticsearch-head插件查看集群情况:

    • 当然,如果只是在相同节点数目的集群上增加更多的副本分片并不能提高性能,因为每个分片从节点上获得的资源会变少。你需要增加更多的硬件资源来提升吞吐量。 但是更多的副本分片数提高了数据冗余量:按照上面的节点配置,我们可以在失去2个节点的情况下不丢失任何数据。

4.3.4 应对故障

  • 我们关闭第一个节点,这时集群的状态为:关闭了一个节点后的集群。

    1645253770313
  • 我们关闭的节点是一个主节点。而集群必须拥有一个主节点来保证正常工作,所以发生的第一件事情就是选举一个新的主节点:Node2。在我们关闭Node1的同时也失去了主分片1和2,并且在缺失主分片的时候索引也不能正常工作。如果此时来检查集群的状况,我们看到的状态将会为red:不是所有主分片都在正常工作。

  • 幸运的是,在其它节点上存在着这两个主分片的完整副本,所以新的主节点立即将这些分片在Node2和Node3上对应的副本分片提升为主分片, 此时集群的状态将会为yellow。这个提升主分片的过程是瞬间发生的,如同按下一个开关一般。

  • 为什么集群状态是yellow而不是green呢?

    • 虽然我们拥有所有的三个主分片,但是同时设置了每个主分片需要对应2份副本分片,而此时只存在一份副本分片。 所以集群不能为green的状态,不过我们不必过于担心:如果我们同样关闭了Node2,我们的程序依然可以保持在不丢任何数据的情况下运行,因为Node3为每一个分片都保留着一份副本。

    • 如果想回复原来的样子,要确保Node-1的配置文件有如下配置:

      1
      2
      # 候选主节点的地址,在开启服务后可以被选为主节点
      discovery.seed_hosts: ["localhost:9302", "localhost:9303"]

      集群可以将缺失的副本分片再次进行分配,那么集群的状态也将恢复成之前的状态。 如果Node 1依然拥有着之前的分片,它将尝试去重用它们,同时仅从主分片复制发生了修改的数据文件。和之前的集群相比,只是Master节点切换了。

4.4 路由计算

  • 当索引一个文档的时候,文档会被存储到一个主分片中。Elasticsearch如何知道一个文档应该存放到哪个分片中呢?当我们创建文档时,它如何决定这个文档应当被存储在分片1还是分片2中呢?首先这肯定不会是随机的,否则将来要获取文档的时候我们就不知道从何处寻找了。实际上,这个过程是根据下面这个公式决定的:

    1
    shard = hash(routing) % number_of_primary_shards
    • routing是一个可变值,默认是文档的_id,也可以设置成一个自定义的值。routing通过hash函数生成一个数字,然后这个数字再除以number_of_primary_shards(主分片的数量)后得到余数 。这个分布在0number_of_primary_shards-1之间的余数,就是我们所寻求的文档所在分片的位置。

    • 这就解释了为什么我们要在创建索引的时候就确定好主分片的数量并且永远不会改变这个数量,因为如果数量变化了,那么所有之前路由的值都会无效,文档也再也找不到了。

    • 所有的文档API(get、index、delete、bulk、update以及mget)都接受一个叫做routing的路由参数,通过这个参数我们可以自定义文档到分片的映射。一个自定义的路由参数可以用来确保所有相关的文档—一例如所有属于同一个用户的文档——都被存储到同一个分片中。

4.5 分片控制

  • 我们可以发送请求到集群中的任一节点。每个节点都有能力处理任意请求。每个节点都知道集群中任一文档位置,所以可以直接将请求转发到需要的节点上。在下面的例子中,如果将所有的请求发送到Node1001,我们将其称为协调节点coordinating node

  • 当发送请求的时候, 为了扩展负载,更好的做法是轮询集群中所有的节点。

4.5.1 数据写流程

  • 新建、索引和删除请求都是写操作,必须在主分片上面完成之后才能被复制到相关的副本分片。

    1645256406040
    • 客户端向Node1发送新建、索引或者删除请求。
    • 节点使用文档的_id确定文档属于分片0。请求会被转发到Node3,因为分片0的主分片目前被分配在Node3上。
    • Node3在主分片上面执行请求。如果成功了,它将请求并行转发到Node1Node2的副本分片上。一旦所有的副本分片都报告成功,Node3将向协调节点报告成功,协调节点向客户端报告成功。
  • 在客户端收到成功响应时,文档变更已经在主分片和所有副本分片执行完成,变更是安全的。 有一些可选的请求参数允许您影响这个过程,可能以数据安全为代价提升性能。这些选项很少使用,因为Elasticsearch已经很快,但是为了完整起见,请参考下面表格:

    参数 含义
    consistency consistency,即一致性。在默认设置下,即使仅仅是在试图执行一个_写_操作之 前,主分片都会要求 必须要有 规定数量(quorum)(或者换种说法,也即必须要 有大多数)的分片副本处于活跃可用状态,才会去执行_写_操作(其中分片副本 可以是主分片或者副本分片)。这是为了避免在发生网络分区故障(network partition)的时候进行_写_操作,进而导致数据不一致。规定数量即:int( (primary + number_of_replicas) / 2 ) + 1
    consistency参数的值可以设为one (只要主分片状态ok就允许执行_写_操 作),all(必须要主分片和所有副本分片的状态没问题才允许执行_写_操作), 或quorum。默认值为quorum , 即大多数的分片副本状态没问题就允许执行_写_ 操作。
    注意,规定数量的计算公式中number_of_replicas指的是在索引设置中的设定副本分片数,而不是指当前处理活动状态的副本分片数。如果你的索引设置中指定了当前索引拥有三个副本分片,那规定数量的计算结果即:int( (primary + 3 replicas) / 2 ) + 1 = 3
    如果此时你只启动两个节点,那么处于活跃状态的分片副本数量就达不到规定数量,也因此您将无法索引和删除任何文档。
    timeout 如果没有足够的副本分片会发生什么? Elasticsearch会等待,希望更多的分片出现。默认情况下,它最多等待1分钟。 如果你需要,你可以使用timeout参数使它更早终止,100是100毫秒,30s是30秒。
  • 新索引默认有1个副本分片,这意味着为满足规定数量应该需要两个活动的分片副本。 但是,这些默认的设置会阻止我们在单一节点上做任何事情。为了避免这个问题,要求只有当number_of_replicas大于1的时候,规定数量才会执行。

4.5.2 数据读流程

  • 我们可以从主分片或者从其它任意副本分片检索文档。

    1645257077200 1645257112246
    • 客户端向Node1发送获取请求。
    • 节点使用文档的_id来确定文档属于分片0。分片0的副本分片存在于所有的三个节点上。在这种情况下,它将请求转发到Node2
    • Node2将文档返回给Node1,然后将文档返回给客户端。
  • 在处理读取请求时,协调结点在每次请求的时候都会通过轮询所有的副本分片来达到负载均衡。在文档被检索时,已经被索引的文档可能已经存在于主分片上但是还没有复制到副本分片。 在这种情况下,副本分片可能会报告文档不存在,但是主分片可能成功返回文档。一旦索引请求成功返回给用户,文档在主分片和副本分片都是可用的。

4.5.3 数据更新流程

  • 部分更新一个文档结合了先前说明的读取和写入流程:

    1645257315478
    • 客户端向Node1发送更新请求。
    • 它将请求转发到主分片所在的Node3
    • Node3从主分片检索文档,修改_source字段中的JSON,并且尝试重新索引主分片的文档。如果文档已经被另一个进程修改,它会重试步骤3,超过retry_on_conflict次后放弃。
    • 如果Node3成功地更新文档,它将新版本的文档并行转发到Node1Node2上的副本分片,重新建立索引。一旦所有副本分片都返回成功,Node3向协调节点也返回成功,协调节点向客户端返回成功。
  • 当主分片把更改转发到副本分片时,它不会转发更新请求。相反,它转发完整文档的新版本。请记住,这些更改将会异步转发到副本分片,并且不能保证它们以发送它们相同的顺序到达。 如果Elasticsearch仅转发更改请求,则可能以错误的顺序应用更改,导致得到损坏的文档。

4.5.4 多文档操作流程

  • mgetbulk API的模式类似于单文档模式。区别在于协调节点知道每个文档存在于哪个分片中。它将整个多文档请求分解成每个分片的多文档请求,并且将这些请求并行转发到每个参与节点。

  • 协调节点一旦收到来自每个节点的应答,就将每个节点的响应收集整理成单个响应,返回给客户端。

    1645257643017
    • 客户端向Node1发送mget请求。
    • Node1为每个分片构建多文档获取请求,然后并行转发这些请求到托管在每个所需的主分片或者副本分片的节点上。一旦收到所有答复,Node1构建响应并将其返回给客户端。
  • 可以对docs数组中每个文档设置routing参数。

  • bulk API,允许在单个批量请求中执行多个创建、索引、删除和更新请求。

    1645257766010
    • 客户端向Node1发送bulk请求。
    • Node1为每个节点创建一个批量请求,并将这些请求并行转发到每个包含主分片的节点主机。
    • 主分片一个接一个按顺序执行每个操作。当每个操作成功时,主分片并行转发新文档(或删除)到副本分片,然后执行下一个操作。一旦所有的副本分片报告所有操作成功, 该节点将向协调节点报告成功,协调节点将这些响应收集整理并返回给客户端。

4.6 分片原理

  • 分片是Elasticsearch最小的工作单元。但是究竟什么是一个分片,它是如何工作的?
    • 传统的数据库每个字段存储单个值,但这对全文检索并不够。文本字段中的每个单词需要被搜索,对数据库意味着需要单个字段有索引多值的能力。最好的支持是一个字段多个值 需求的数据结构是倒排索引。

4.6.1 倒排索引

  • Elasticsearch使用一种称为倒排索引的结构,它适用于快速的全文搜索。

  • 见其名,知其意,有倒排索引,肯定会对应有正向索引。正向索引(forward index), 反向索引(inverted index),更熟悉的名字是倒排索引。

  • 所谓的正向索引,就是搜索引擎会将待搜索的文件都对应一个文件ID,搜索时将这个ID和搜索关键字进行对应,形成K-V对,然后对关键字进行统计计数。

  • 但是互联网上收录在搜索引擎中的文档的数目是个天文数字,这样的索引结构根本无法满足实时返回排名结果的要求。所以,搜索引擎会将正向索引重新构建为倒排索引,即把文件ID对应到关键词的映射转换为关键词到文件ID的映射,每个关键词都对应着一系列的文件, 这些文件中都出现这个关键词。

倒排索引举例

  • 一个倒排索引由文档中所有不重复词的列表构成,对于其中每个词,有一个包含它的文档列表。例如,假设我们有两个文档,每个文档的content域包含如下内容:

    • The quick brown fox jumped over the lazy dog
    • Quick brown foxes leap over lazy dogs in summer
  • 为了创建倒排索引,我们首先将每个文档的content域拆分成单独的词(我们称它为词条或tokens),创建一个包含所有不重复词条的排序列表,然后列出每个词条出现在哪个文档。结果如下所示:

  • 现在,如果我们想搜索quick brown,我们只需要查找包含每个词条的文档:

    • 两个文档都匹配,但是第一个文档比第二个匹配度更高。如果我们使用仅计算匹配词条数量的简单相似性算法,那么我们可以说,对于我们查询的相关性来讲,第一个文档比第二个文档更佳。
  • 但是,我们目前的倒排索引有一些问题:

    • Quickquick以独立的词条出现,然而用户可能认为它们是相同的词。

    • foxfoxes非常相似,就像dogdogs,他们有相同的词根。

    • jumpedleap,尽管没有相同的词根,但他们的意思很相近。他们是同义词。

    • 使用前面的索引搜索+Quick +fox不会得到任何匹配文档。(记住,+前缀表明这个词必须存在)只有同时出现Quickfox的文档才满足这个查询条件,但是第一个文档包含quick fox,第二个文档包含Quick foxes

    • 我们的用户可以合理的期望两个文档与查询匹配。我们可以做的更好。如果我们将词条规范为标准模式,那么我们可以找到与用户搜索的词条不完全一致,但具有足够相关性的文档。例如:

      • Quick可以小写化为quick
      • foxes可以词干提取–变为词根的格式–为fox。类似的,dogs可以为提取为dog
      • jumpedleap是同义词,可以索引为相同的单词jump
    • 现在索引看上去像这样:

      • 这还远远不够。我们搜索+Quick +fox仍然会失败,因为在我们的索引中,已经没有Quick了。但是,如果我们对搜索的字符串使用与content域相同的标准化规则,会变成查询+quick +fox,这样两个文档都会匹配!分词和标准化的过程称为分析。 这非常重要。你只能搜索在索引中出现的词条,所以索引文本和查询字符串必须标准化为相同的格式。

4.6.2 文档搜索

  • 早期的全文检索会为整个文档集合建立一个很大的倒排索引并将其写入到磁盘。一旦新的索引就绪,旧的就会被其替换,这样最近的变化便可以被检索到。
  • 倒排索引被写入磁盘后是不可改变的。不变性有重要的价值:
    • 不需要锁。如果你从来不更新索引,你就不需要担心多进程同时修改数据的问题。
    • 一旦索引被读入内核的文件系统缓存,便会留在哪里,由于其不变性。只要文件系统缓存中还有足够 的空间,那么大部分读请求会直接请求内存,而不会命中磁盘。这提供了很大的性能提升。
    • 其它缓存(像filter缓存),在索引的生命周期内始终有效。它们不需要在每次数据改变时被重建,因为 数据不会变化。
    • 写入单个大的倒排索引允许数据被压缩,减少磁盘I/O和需要被缓存到内存的索引的使用量。
  • 当然,一个不变的索引也有不好的地方。主要事实是它是不可变的! 你不能修改它。如 果你需要让一个新的文档 可被搜索,你需要重建整个索引。这要么对一个索引所能包含的 数据量造成了很大的限制,要么对索引可被更新的频率造成了很大的限制。

4.6.3 动态更新索引

  • 如何在保留不变性的前提下实现倒排索引的更新?

    • 用更多的索引。通过增加新的补充索引来反映新近的修改,而不是直接重写整个倒排索引。每一个倒排索引都会被轮流查询到,从最早的开始查询完后再对结果进行合并。

    • Elasticsearch基于Lucene,这个java库引入了按段搜索的概念。每一段本身都是一个倒排索引,但索引在Lucene中除表示所有段的集合外,还增加了提交点的概念,即一个列出了所有已知段的文件。

      1645263465639
    • 按段搜索会以如下流程执行:

      • (1)新文档被收集到内存索引缓存。

        1645260037858
      • (2)不时地,缓存被提交。

        • ①一个新的段——一个追加的倒排索引被写入磁盘。
        • ②一个新的包含新段名字的提交点被写入磁盘。
        • ③磁盘进行同步——所有在文件系统缓存中等待的写入都刷新到磁盘,以确保它们被写入物理文件。
      • (3)新的段被开启,让它包含的文档可见以被搜索。

      • (4)内存缓存被清空,等待接收新的文档。

        1645260216982
    • 当一个查询被触发,所有已知的段按顺序被查询。词项统计会对所有段的结果进行聚合,以保证每个词和每个文档的关联都被准确计算。 这种方式可以用相对较低的成本将新文档添加到索引。

    • 段是不可改变的,所以既不能从把文档从旧的段中移除,也不能修改旧的段来进行反映文档的更新。取而代之的是,每个提交点会包含一个.del文件,文件中会列出这些被删除文档的段信息。

    • 当一个文档被“删除”时,它实际上只是在.del文件中被标记删除。一个被标记删除的文档仍然可以被查询匹配到, 但它会在最终结果被返回前从结果集中移除。

    • 文档更新也是类似的操作方式:当一个文档被更新时,旧版本文档被标记删除,文档的新版本被索引到一个新的段中。 可能两个版本的文档都会被一个查询匹配到,但被删除的那个旧版本文档在结果集返回前就已经被移除。

4.6.4 近实时搜索

  • 随着按段(per-segment)搜索的发展,一个新的文档从索引到可被搜索的延迟显著降低了。新文档在几分钟之内即可被检索,但这样还是不够快。磁盘在这里成为了瓶颈。提交 (Commiting)一个新的段到磁盘需要一个fsync来确保段被物理性地写入磁盘,这样在断电的时候就不会丢失数据。但是fsync操作代价很大,如果每次索引一个文档都去执行一次的话会造成很大的性能问题。

  • 我们需要的是一个更轻量的方式来使一个文档可被搜索,这意味着fsync要从整个过程中被移除。在Elasticsearch和磁盘之间是文件系统缓存。像之前描述的一样,在内存索引缓冲区中的文档会被写入到一个新的段中。 但是这里新段会被先写入到文件系统缓存——这一步代价会比较低,稍后再被刷新到磁盘——这一步代价比较高。不过只要文件已经在缓存中, 就可以像其它文件一样被打开和读取了。

    1645260547846
  • Lucene允许新段被写入和打开—使其包含的文档在未进行一次完整提交时便对搜索可见。这种方式比进行一次提交代价要小得多,并且在不影响性能的前提下可以被频繁地执行。

    1645261295332
  • Elasticsearch中,写入和打开一个新段的轻量的过程叫做refresh。 默认情况下每个分片会每秒自动刷新一次。这就是为什么我们说Elasticsearch是近实时搜索: 文档的变化并不是立即对搜索可见,但会在一秒之内变为可见。

  • 这些行为可能会对新用户造成困惑:他们索引了一个文档然后尝试搜索它,但却没有搜到。这个问题的解决办法是用refresh API执行一次手动刷新:/users/_refresh

  • 尽管刷新是比提交轻量很多的操作,它还是会有性能开销。当写测试的时候,手动刷新很有用,但是不要在生产环境下每次索引一个文档都去手动刷新。相反,你的应用需要意识到Elasticsearch的近实时的性质,并接受它的不足。

  • 并不是所有的情况都需要每秒刷新。可能你正在使用Elasticsearch索引大量的日志文件,你可能想优化索引速度而不是近实时搜索, 可以通过设置refresh_interval, 降低每个索引的刷新频率。

    1
    2
    3
    4
    5
    {
    "settings": {
    "refresh_interval": "30s"
    }
    }
  • refresh_interval可以在既存索引上进行动态更新。在生产环境中,当你正在建立一个大的新索引时,可以先关闭自动刷新,待开始使用该索引时,再把它们调回来。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    # 关闭自动刷新
    PUT /users/_settings
    {
    "refresh_interval": -1
    }
    # 每一秒刷新
    PUT /users/_settings
    {
    "refresh_interval": "1s"
    }

4.6.5 持久化变更

  • 如果没有用fsync把数据从文件系统缓存刷(flush)到硬盘,我们不能保证数据在断电甚至是程序正常退出之后依然存在。为了保证Elasticsearch的可靠性,需要确保数据变化被持久化到磁盘。在动态更新索引,我们说一次完整的提交会将段刷到磁盘,并写入一个包含所有段列表的提交点。Elasticsearch在启动或重新打开一个索引的过程中使用这个提交点来判断哪些段隶属于当前分片。

  • 即使通过每秒刷新(refresh)实现了近实时搜索,我们仍然需要经常进行完整提交来确保能从失败中恢复。但在两次提交之间发生变化的文档怎么办?我们也不希望丢失掉这些数据。Elasticsearch增加了一个translog,或者叫事务日志,在每一次对Elasticsearch进行操作时均进行了日志记录。

  • 整个流程如下:

    • 一个文档被索引之后,就会被添加到内存缓冲区,并且追加到了translog

      1645262068695
    • 刷新(refresh)使分片每秒被刷新(refresh)一次。

      • 这些在内存缓冲区的文档被写入到一个新的段中,且没有进行fsync操作。
      • 这个段被打开,使其可被搜索。
      • 内存缓冲区被清空。
      1645262160950
    • 这个进程继续工作,更多的文档被添加到内存缓冲区和追加到事务日志。

      1645262187894
    • 每隔一段时间,例如translog变得越来越大,索引被刷新(flush);一个新的translog被创建,并且一个全量提交被执行。

      • 所有在内存缓冲区的文档都被写入一个新的段。
      • 缓冲区被清空。
      • 一个提交点被写入硬盘。
      • 文件系统缓存通过fsync被刷新(flush)。
      • 老的translog 删除。
  • translog提供所有还没有被刷到磁盘的操作的一个持久化纪录。当Elasticsearch启动的时候, 它会从磁盘中使用最后一个提交点去恢复已知的段,并且会重放translog中所有在最后一次提交后发生的变更操作。

  • translog也被用来提供实时CRUD。当你试着通过ID查询、更新、删除一个文档,它会在尝试从相应的段中检索之前, 首先检查translog任何最近的变更。这意味着它总是能够实时地获取到文档的最新版本。

    1645262478549
  • 执行一个提交并且截断translog的行为在Elasticsearch被称作一次flush,分片每30分钟被自动刷新(flush),或者在translog太大的时候也会刷新。

  • 你很少需要自己手动执行flush操作;通常情况下,自动刷新就足够了。这就是说,在重启节点或关闭索引之前执行flush有益于你的索引。当Elasticsearch尝试恢复或重新打开一个索引,它需要重放translog中所有的操作,所以如果日志越短,恢复越快。

  • translog的目的是保证操作不会丢失,在文件被fsync到磁盘前,被写入的文件在重启之后就会丢失。默认 translog是每5秒被fsync刷新到硬盘, 或者在每次写请求完成之后执行(e.g. index, delete, update, bulk)。这个过程在主分片和复制分片都会发生。最终, 基本上,这意味着在整个请求被fsync到主分片和复制分片的translog之前,你的客户端不会得到一个200OK 响应。

  • 在每次请求后都执行一个fsync会带来一些性能损失,尽管实践表明这种损失相对较小(特别是bulk导入,它在一次请求中平摊了大量文档的开销)。

  • 但是对于一些大容量的偶尔丢失几秒数据问题也并不严重的集群,使用异步的fsync还是比较有益的。比如,写入的数据被缓存到内存中,再每5秒执行一次fsync。如果你决定使用异步translog的话,你需要保证在发生crash时,丢失掉sync_interval时间段的数据也无所谓。请在决定前知晓这个特性。如果你不确定这个行为的后果,最好是使用默认的参数( "index.translog.durability": "request" )来避免数据丢失。

4.6.6 段合并

  • 由于自动刷新流程每秒会创建一个新的段 ,这样会导致短时间内的段数量暴增。而段数目太多会带来较大的麻烦。 每一个段都会消耗文件句柄、内存和cpu运行周期。更重要的是,每个搜索请求都必须轮流检查每个段;所以段越多,搜索也就越慢。

  • Elasticsearch通过在后台进行段合并来解决这个问题。小的段被合并到大的段,然后这些大的段再被合并到更大的段。

  • 段合并的时候会将那些旧的已删除文档从文件系统中清除。被删除的文档(或被更新文档的旧版本)不会被拷贝到新的大段中。

  • 启动段合并不需要你做任何事。进行索引和搜索时会自动进行。

    • 当索引的时候,刷新(refresh)操作会创建新的段并将段打开以供搜索使用。

    • 合并进程选择一小部分大小相似的段,并且在后台将它们合并到更大的段中。这并不会中断索引和搜索。

      1645262905393
    • 一旦合并结束,老的段被删除。

      • 新的段被刷新(flush)到了磁盘。写入一个包含新段且排除旧的和较小的段的新提交点。
      • 新的段被打开用来搜索。
      • 老的段被删除。
      1645262992127
  • 合并大的段需要消耗大量的I/O和CPU资源,如果任其发展会影响搜索性能。Elasticsearch在默认情况下会对合并流程进行资源限制,所以搜索仍然有足够的资源很好地执行。

4.6.7 总结

4.7 文档分析

  • 分析包含下面的过程:
    • 将一块文本分成适合于倒排索引的独立的词条。
    • 将这些词条统一化为标准格式以提高它们的“可搜索性”,或者recall
  • 分析器执行上面的工作。分析器实际上是将三个功能封装到了一个包里:
    • 字符过滤器:首先,字符串按顺序通过每个字符过滤器 。他们的任务是在分词前整理字符串。一个字符过滤器可以用来去掉HTML,或者将&转化成and
    • 分词器:其次,字符串被分词器分为单个的词条。一个简单的分词器遇到空格和标点的时候, 可能会将文本拆分成词条。
    • Token过滤器:最后,词条按顺序通过每个token过滤器 。这个过程可能会改变词条(例如,小写化 Quick),删除词条(例如, 像a,and,the等无用词),或者增加词条(例如,像jumpleap这种同义词)。

4.7.1 内置分析器

  • Elasticsearch还附带了可以直接使用的预包装的分析器。接下来我们会列出最重要的分析器。为了证明它们的差异,我们看看每个分析器会从下面的字符串得到哪些词条:Set the shape to semi-transparent by calling set_trans(5)
    • 标准分析器: 标准分析器是Elasticsearch默认使用的分析器。它是分析各种语言文本最常用的选择。它根据Unicode联盟定义的单词边界划分文本。删除绝大部分标点。最后,将词条小写。 它会产生:set, the, shape, to, semi, transparent, by, calling, set_trans, 5
    • 简单分析器:简单分析器在任何不是字母的地方分隔文本,将词条小写。它会产生:set, the, shape, to, semi, transparent, by, calling, set, trans
    • 空格分析器:空格分析器在空格的地方划分文本。它会产生:Set, the, shape, to, semi-transparent, by, calling, set_trans(5)
    • 语言分析器:特定语言分析器可用于 很多语言。它们可以考虑指定语言的特点。例如, 英语分析器附带了一组英语无用词(常用单词,例如and或者the,它们对相关性没有多少影响),它们会被删除。由于理解英语语法的规则,这个分词器可以提取英语单词的词干。英语分词器会产生下面的词条:set, shape, semi, transpar, call, set_tran, 5(注意看transparent、calling和set_trans已经变为词根格式)。

4.7.2 分析器使用场景

  • 当我们索引一个文档,它的全文域被分析成词条以用来创建倒排索引。 但是,当我们在全文域搜索的时候,我们需要将查询字符串通过相同的分析过程 ,以保证我们搜索的词条格式与索引中的词条格式一致。
  • 全文查询,理解每个域是如何定义的,因此它们可以做正确的事:
    • 当你查询一个全文域时,会对查询字符串应用相同的分析器,以产生正确的搜索词条列表。
    • 当你查询一个精确值域时,不会分析查询字符串,而是搜索你指定的精确值。

4.7.3 测试分析器

  • 有些时候很难理解分词的过程和实际被存储到索引中的词条,特别是你刚接触Elasticsearch。为了理解发生了什么,你可以使用analyze API来看文本是如何被分析的。在消息体里,指定分析器和要分析的文本:

    1
    2
    3
    4
    5
    6
    7
    8
    # 发送GET请求
    http://localhost:9200/_analyze

    # 请求体如下
    {
    "analyzer": "standard",
    "text": "Text to analyze"
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      {
      "tokens": [
      {
      "token": "text",
      "start_offset": 0,
      "end_offset": 4,
      "type": "<ALPHANUM>",
      "position": 1
      },
      {
      "token": "to",
      "start_offset": 5,
      "end_offset": 7,
      "type": "<ALPHANUM>",
      "position": 2
      },
      {
      "token": "analyze",
      "start_offset": 8,
      "end_offset": 15,
      "type": "<ALPHANUM>",
      "position": 3
      }
      ]
      }

      token是实际存储到索引中的词条。position指明词条在原始文本中出现的位置。start_offsetend_offset指明字符在原始字符串中的位置。

4.7.4 指定分析器

  • 当Elasticsearch在你的文档中检测到一个新的字符串域,它会自动设置其为一个全文字符串域,使用标准分析器对它进行分析。你不希望总是这样。可能你想使用一个不同的分析器,适用于你的数据使用的语言。有时候你想要一个字符串域就是一个字符串域—不使用分析,直接索引你传入的精确值,例如用户ID或者一个内部的状态域或标签。要做到这一点,我们必须手动指定这些域的映射。

4.7.5 IK分词器

  • 测试以下例子:

    1
    2
    3
    4
    # GET http://localhost:9200/_analyze
    {
    "text":"测试单词"
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      {
      "tokens": [
      {
      "token": "测",
      "start_offset": 0,
      "end_offset": 1,
      "type": "<IDEOGRAPHIC>",
      "position": 0
      },
      {
      "token": "试",
      "start_offset": 1,
      "end_offset": 2,
      "type": "<IDEOGRAPHIC>",
      "position": 1
      },
      {
      "token": "单",
      "start_offset": 2,
      "end_offset": 3,
      "type": "<IDEOGRAPHIC>",
      "position": 2
      },
      {
      "token": "词",
      "start_offset": 3,
      "end_offset": 4,
      "type": "<IDEOGRAPHIC>",
      "position": 3
      }
      ]
      }

      ES的默认分词器无法识别中文中测试、单词这样的词汇,而是简单的将每个字拆完分为一个词。这样的结果显然不符合我们的使用要求,所以我们需要下载ES对应版本的中文分词器。

    • 这里采用IK中文分词器,下载地址为: https://github.com/medcl/elasticsearch-analysis-ik/releases/tag/v7.8.0。将解压后的后的文件夹放入`ES`根目录下的`plugins`目录下,重启`ES`即可使用。

      1
      2
      3
      4
      5
      # GET http://localhost:9200/_analyze
      {
      "text":"测试单词",
      "analyzer":"ik_max_word"
      }
      • ik_max_word:会将文本做最细粒度的拆分。

      • ik_smart:会将文本做最粗粒度的拆分。

      • 返回结果:

        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        11
        12
        13
        14
        15
        16
        17
        18
        {
        "tokens": [
        {
        "token": "测试",
        "start_offset": 0,
        "end_offset": 2,
        "type": "CN_WORD",
        "position": 0
        },
        {
        "token": "单词",
        "start_offset": 2,
        "end_offset": 4,
        "type": "CN_WORD",
        "position": 1
        }
        ]
        }
    • ES中也可以进行扩展词汇,首先查询:

      1
      2
      3
      4
      5
      # GET http://localhost:9200/_analyze
      {
      "text":"弗雷尔卓德",
      "analyzer":"ik_max_word"
      }

      仅仅可以得到每个字的分词结果,我们需要做的就是使分词器识别到弗雷尔卓德也是一个词语。

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      33
      34
      35
      36
      37
      38
      39
      {
      "tokens": [
      {
      "token": "弗",
      "start_offset": 0,
      "end_offset": 1,
      "type": "CN_CHAR",
      "position": 0
      },
      {
      "token": "雷",
      "start_offset": 1,
      "end_offset": 2,
      "type": "CN_CHAR",
      "position": 1
      },
      {
      "token": "尔",
      "start_offset": 2,
      "end_offset": 3,
      "type": "CN_CHAR",
      "position": 2
      },
      {
      "token": "卓",
      "start_offset": 3,
      "end_offset": 4,
      "type": "CN_CHAR",
      "position": 3
      },
      {
      "token": "德",
      "start_offset": 4,
      "end_offset": 5,
      "type": "CN_CHAR",
      "position": 4
      }
      ]
      }

      首先进入ES根目录中的plugins文件夹下的ik文件夹,进入config目录,创建custom.dic文件,写入弗雷尔卓德。同时打开IKAnalyzer.cfg.xml文件,将新建的custom.dic配置其中,重启ES服务器。

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      <?xml version="1.0" encoding="UTF-8"?>
      <!DOCTYPE properties SYSTEM "http://java.sun.com/dtd/properties.dtd">
      <properties>
      <comment>IK Analyzer 扩展配置</comment>
      <!--用户可以在这里配置自己的扩展字典 -->
      <entry key="ext_dict">custom.dic</entry>
      <!--用户可以在这里配置自己的扩展停止词字典-->
      <entry key="ext_stopwords"></entry>
      <!--用户可以在这里配置远程扩展字典 -->
      <!-- <entry key="remote_ext_dict">words_location</entry> -->
      <!--用户可以在这里配置远程扩展停止词字典-->
      <!-- <entry key="remote_ext_stopwords">words_location</entry> -->
      </properties>

      再次查询后返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      {
      "tokens": [
      {
      "token": "弗雷尔卓德",
      "start_offset": 0,
      "end_offset": 5,
      "type": "CN_WORD",
      "position": 0
      }
      ]
      }

4.7.6 自定义分析器

  • 虽然Elasticsearch带有一些现成的分析器,然而在分析器上Elasticsearch真正的强大之处在于,你可以通过在一个适合你的特定数据的设置之中组合字符过滤器、分词器、词汇单 元过滤器来创建自定义的分析器。在分析与分析器我们说过,一个分析器就是在一个包里面组合了三种函数的一个包装器, 三种函数按照顺序被执行:
    • 字符过滤器:字符过滤器用来整理一个尚未被分词的字符串。例如,如果我们的文本是HTML格式的,它会包含像<p>或者<div>这样的HTML标签,这些标签是我们不想索引的。我们可以使用html清除字符过滤器来移除掉所有的HTML标签,并且像把&Aacute;转换为相对应的Unicode字符Á这样,转换HTML实体。一个分析器可能有0个或者多个字符过滤器。
    • 分词器:一个分析器必须有一个唯一的分词器。分词器把字符串分解成单个词条或者词汇单元。标准分析器里使用的标准分词器把一个字符串根据单词边界分解成单个词条,并且移除掉大部分的标点符号,然而还有其他不同行为的分词器存在。例如,关键词分词器完整地输出接收到的同样的字符串,并不做任何分词。空格分词器只根据空格分割文本。正则分词器根据匹配正则表达式来分割文本。
    • 词单元过滤器:经过分词,作为结果的词单元流会按照指定的顺序通过指定的词单元过滤器。词单元过滤器可以修改、添加或者移除词单元。我们已经提到过lowercasestop词过滤器,但是在Elasticsearch里面还有很多可供选择的词单元过滤器。词干过滤器把单词遏制为词干。ascii_folding过滤器移除变音符,把一个像"très”这样的词转换为“tres”ngramedge_ngram词单元过滤器可以产生适合用于部分匹配或者自动补全的词单元。

自定义分析器例子

  • 发送以下请求:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    # PUT http://localhost:9200/my_index
    {
    "settings": {
    "analysis": {
    "char_filter": {
    "&_to_and": {
    "type": "mapping",
    "mappings": [
    "&=> and "
    ]
    }
    },
    "filter": {
    "my_stopwords": {
    "type": "stop",
    "stopwords": [
    "the",
    "a"
    ]
    }
    },
    "analyzer": {
    "my_analyzer": {
    "type": "custom",
    "char_filter": [
    "html_strip",
    "&_to_and"
    ],
    "tokenizer": "standard",
    "filter": [
    "lowercase",
    "my_stopwords"
    ]
    }
    }
    }
    }
    }
  • 索引被创建以后,使用analyze API来测试这个新的分析器:

    1
    2
    3
    4
    5
    # GET http://127.0.0.1:9200/my_index/_analyze
    {
    "text":"The quick & brown fox",
    "analyzer": "my_analyzer"
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      20
      21
      22
      23
      24
      25
      26
      27
      28
      29
      30
      31
      32
      {
      "tokens": [
      {
      "token": "quick",
      "start_offset": 4,
      "end_offset": 9,
      "type": "<ALPHANUM>",
      "position": 1
      },
      {
      "token": "and",
      "start_offset": 10,
      "end_offset": 11,
      "type": "<ALPHANUM>",
      "position": 2
      },
      {
      "token": "brown",
      "start_offset": 12,
      "end_offset": 17,
      "type": "<ALPHANUM>",
      "position": 3
      },
      {
      "token": "fox",
      "start_offset": 18,
      "end_offset": 21,
      "type": "<ALPHANUM>",
      "position": 4
      }
      ]
      }

4.8 文档处理

4.8.1 文档冲突

  • 当我们使用index API更新文档,可以一次性读取原始文档,做我们的修改,然后重新索引整个文档。最近的索引请求将获胜:无论最后哪一个文档被索引,都将被唯一存储在Elasticsearch中。如果其他人同时更改这个文档,他们的更改将丢失。

  • 很多时候这是没有问题的。也许我们的主数据存储是一个关系型数据库,我们只是将数据复制到Elasticsearch中并使其可被搜索。也许两个人同时更改相同的文档的几率很小。或者对于我们的业务来说偶尔丢失更改并不是很严重的问题。

  • 但有时丢失了一个变更就是非常严重的。试想我们使用Elasticsearch存储我们网上商城商品库存的数量,每次我们卖一个商品的时候,我们在Elasticsearch中将库存数量减少。有一天,管理层决定做一次促销。突然地,我们一秒要卖好几个商品。假设有两个web程序并行运行,每一个都同时处理所有商品的销售。

    1645275310917
    • web_1stock_count所做的更改已经丢失,因为web_2不知道它的stock_count的拷贝已经过期。结果我们会认为有超过商品的实际数量的库存,因为卖给顾客的库存商品并不存在,我们将让他们非常失望。
  • 变更越频繁,读数据和更新数据的间隙越长,也就越可能丢失变更。

  • 在数据库领域中,有两种方法通常被用来确保并发更新时变更不会丢失:

    • 悲观并发控制:这种方法被关系型数据库广泛使用,它假定有变更冲突可能发生,因此阻塞访问资源以防止冲突。 一个典型的例子是读取一行数据之前先将其锁住,确保只有放置锁的线程能够对这行数据进行修改。
    • 乐观并发控制:Elasticsearch中使用的这种方法假定冲突是不可能发生的,并且不会阻塞正在尝试的操作。 然而,如果源数据在读写当中被修改,更新将会失败。应用程序接下来将决定该如何解决冲突。 例如,可以重试更新、使用新的数据、或者将相关情况报告给用户。

4.8.2 乐观并发控制

  • Elasticsearch是分布式的。当文档创建、更新或删除时,新版本的文档必须复制到集群中的其他节点。Elasticsearch也是异步和并发的,这意味着这些复制请求被并行发送,并且到达目的地时也许顺序是乱的。Elasticsearch需要一种方法确保文档的旧版本不会覆盖新的版本。

  • 当我们之前讨论index , GET和DELETE请求时,我们指出每个文档都有一个_version(版本号),当文档被修改时版本号递增。Elasticsearch使用这个version号来确保变更以正确顺序得到执行。如果旧版本的文档在新版本之后到达,它可以被简单的忽略。

  • 我们可以利用version号来确保应用中相互冲突的变更不会导致数据丢失。我们通过指定想要修改文档的 version号来达到这个目的。如果该版本不是当前版本号,我们的请求将会失败。

  • 老的版本es使用version,但是新版本不支持了,会报下面的错误,提示我们用if_seq _noif _primary_term

    • 创建文档:

      1
      2
      3
      4
      5
      6
      # PUT http://127.0.0.1:9200/shopping/_create/1001
      {
      "title":"小米手机",
      "category":"小米",
      "price":3999.00
      }
      • 返回结果:

        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        11
        12
        13
        14
        {
        "_index": "shopping",
        "_type": "_doc",
        "_id": "1001",
        "_version": 1,
        "result": "created",
        "_shards": {
        "total": 2,
        "successful": 1,
        "failed": 0
        },
        "_seq_no": 29,
        "_primary_term": 1
        }
    • 更新数据:

      1
      2
      3
      4
      5
      6
      # POST http://127.0.0.1:9200/shopping/_update/1001
      {
      "doc":{
      "title":"华为手机"
      }
      }
      • 返回结果:

        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        11
        12
        13
        14
        {
        "_index": "shopping",
        "_type": "_doc",
        "_id": "1001",
        "_version": 2,
        "result": "updated",
        "_shards": {
        "total": 2,
        "successful": 1,
        "failed": 0
        },
        "_seq_no": 30,
        "_primary_term": 1
        }
    • 采用旧版本的防止冲突更新方法失败:

      1
      2
      3
      4
      5
      6
      # POST http://127.0.0.1:9200/shopping/_update/1001?version=1
      {
      "doc":{
      "title":"华为手机2"
      }
      }
      • 返回结果:

        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        11
        12
        13
        {
        "error": {
        "root_cause": [
        {
        "type": "action_request_validation_exception",
        "reason": "Validation Failed: 1: internal versioning can not be used for optimistic concurrency control. Please use `if_seq_no` and `if_primary_term` instead;"
        }
        ],
        "type": "action_request_validation_exception",
        "reason": "Validation Failed: 1: internal versioning can not be used for optimistic concurrency control. Please use `if_seq_no` and `if_primary_term` instead;"
        },
        "status": 400
        }
    • 采用新版本的防止冲突更新方法成功:

      1
      2
      3
      4
      5
      6
      # POST http://127.0.0.1:9200/shopping/_update/1001?if_seq_no=30&if_primary_term=1
      {
      "doc":{
      "title":"华为手机2"
      }
      }
      • 返回结果:

        1
        2
        3
        4
        5
        6
        7
        8
        9
        10
        11
        12
        13
        14
        {
        "_index": "shopping",
        "_type": "_doc",
        "_id": "1001",
        "_version": 3,
        "result": "updated",
        "_shards": {
        "total": 2,
        "successful": 1,
        "failed": 0
        },
        "_seq_no": 31,
        "_primary_term": 1
        }

4.8.3 外部系统版本控制

  • 一个常见的设置是使用其它数据库作为主要的数据存储,使用Elasticsearch做数据检索,这意味着主数据库的所有更改发生时都需要被复制到Elasticsearch,如果多个进程负责这一数据同步,你可能遇到类似于之前描述的并发问题。

  • 如果你的主数据库已经有了版本号,或一个能作为版本号的字段值比如timestamp,那么你就可以在Elasticsearch中通过增加version_type=external到查询字符串的方式重用这些相同的版本号,版本号必须是大于零的整数,且小于9.2E+18,一个Javalong类型的正值。

  • 外部版本号的处理方式和我们之前讨论的内部版本号的处理方式有些不同,Elasticsearch不是检查当前_version和请求中指定的版本号是否相同,而是检查当前_version是否小于指定的版本号。如果请求成功,外部的版本号作为文档的新_version进行存储。

    1
    2
    3
    4
    # POST http://127.0.0.1:9200/shopping/_doc/1001?version=1&version_type=external
    {
    "title":"华为手机"
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      15
      16
      17
      18
      19
      {
      "error": {
      "root_cause": [
      {
      "type": "version_conflict_engine_exception",
      "reason": "[1001]: version conflict, current version [3] is higher or equal to the one provided [1]",
      "index_uuid": "ZAHmw1H1Ty6WQ35ki3_VxA",
      "shard": "0",
      "index": "shopping"
      }
      ],
      "type": "version_conflict_engine_exception",
      "reason": "[1001]: version conflict, current version [3] is higher or equal to the one provided [1]",
      "index_uuid": "ZAHmw1H1Ty6WQ35ki3_VxA",
      "shard": "0",
      "index": "shopping"
      },
      "status": 409
      }
    1
    2
    3
    4
    # POST http://127.0.0.1:9200/shopping/_doc/1001?version=100&version_type=external
    {
    "title":"华为手机"
    }
    • 返回结果:

      1
      2
      3
      4
      5
      6
      7
      8
      9
      10
      11
      12
      13
      14
      {
      "_index": "shopping",
      "_type": "_doc",
      "_id": "1001",
      "_version": 100,
      "result": "updated",
      "_shards": {
      "total": 2,
      "successful": 1,
      "failed": 0
      },
      "_seq_no": 32,
      "_primary_term": 1
      }