干货 | 全方位深度解读 Elasticsearch 分页查询

本文涉及的产品
检索分析服务 Elasticsearch 版,2核4GB开发者规格 1个月
简介: 1、关于 Elasticsearch 分页查询,这几个问题经常被问到问题1:想请问下,一次性获取索引上的某个字段的所有值(100 万左右),除了把 max_result_window 调大 ,还有没有啥方法?问题2:关于 es 的分页,每次拿 20 条展示在前台,然后点击下一页,在查询后面的20条数据,应该要怎么写?问题3:From+size、Scroll、search_after 的本质区别和应用场景分别是什么?

2、 Elasticsearch 支持的三种分页查询方式

From + Size 查询


Search After 查询


Scroll 查询


下面我就三种方式的联系与区别、优缺点、适用场景等展开进行解读。


2.1 From + size 分页查询

2.1.1 From + size 分页查询定义与实战案例

如下基础查询:


GET kibana_sample_data_flights/_search

默认返回前10个匹配的匹配项。其中:


from:未指定,默认值是 0,注意不是1,代表当前页返回数据的起始值。


size:未指定,默认值是 10,代表当前页返回数据的条数。


如下指定条件查询和排序:


GET kibana_sample_data_flights/_search

{

 "from": 0,

 "size":20,

 "query": {

   "match": {

     "DestWeather": "Sunny"

   }

 },

 "sort": [

   {

     "FlightTimeHour": {

       "order": "desc"

     }

   }

 ]

}

共返回 20 条数据。


其中:from + size 两个参数定义了结果页面显示数据的内容。


2.1.2 From + size 查询优缺点及适用场景

From + size 查询优点


支持随机翻页。


From + size 查询缺点


受制于 max_result_window 设置,不能无限制翻页。


存在深度翻页问题,越往后翻页越慢。


From + size 查询适用场景


第一:非常适合小型数据集或者大数据集返回 Top N(N <= 10000)结果集的业务场景。


第二:类似主流 PC 搜索引擎(谷歌、bing、百度、360、sogou等)支持随机跳转分页的业务场景。




2.1.3 深度翻页不推荐使用 From + size

Elasticsearch 会限制最大分页数,避免大数据量的召回导致性能低下。


Elasticsearch 的 max_result_window 默认值是:10000。也就意味着:如果每页有 10 条数据,会最大翻页至 1000 页。


实际主流搜索引擎都翻不了那么多页,举例:百度搜索“上海”,翻到第 76 页,就无法再往下翻页了,提示信息如下截图所示:

image.png

GET kibana_sample_data_flights/_search

{

 "from": 0,

 "size":10001

}

GET kibana_sample_data_flights/_search

{

 "from": 10001,

 "size":10

}

报错如下:


{

 "error" : {

   "root_cause" : [

     {

       "type" : "illegal_argument_exception",

       "reason" : "Result window is too large, from + size must be less than or equal to: [10000] but was [10001]. See the scroll api for a more efficient way to request large data sets. This limit can be set by changing the [index.max_result_window] index level setting."

     }

   ],

什么原因?超过了最大窗口的限制,index.max_result_window 默认值为10000。


报错信息还同时给出了两个解决方案:


方案一:大数据集召回数据使用:scroll api。


后面会详细讲解。


方案二:调大 index.max_result_window 默认值。


PUT kibana_sample_data_flights/_settings

{

   "index.max_result_window":50000

}

官方建议:避免过度使用 from 和 size 来分页或一次请求太多结果。


不推荐使用 from + size 做深度分页查询的核心原因:


搜索请求通常跨越多个分片,每个分片必须将其请求的命中内容以及任何先前页面的命中内容加载到内存中。


对于翻页较深的页面或大量结果,这些操作会显著增加内存和 CPU 使用率,从而导致性能下降或节点故障。


什么意思呢?


GET kibana_sample_data_flights/_search

{

 "from": 10001,

 "size": 10

}

共 10 条数据加载到内存吗?不是的!


共:10011 条数据加载到内存,然后经过后台处理后返回了最后 10 条我们想要的数据。


那也就意味着,越往后翻页(也就是深度翻页)需要加载的数据量越大,势必会越耗费 CPU + 内存资源,响应也会越慢!


2.2 search_after 查询

2.2.1 search_after 查询定义与实战案例

search_after 查询本质:使用前一页中的一组排序值来检索匹配的下一页。


前置条件:使用 search_after 要求后续的多个请求返回与第一次查询相同的排序结果序列。也就是说,即便在后续翻页的过程中,可能会有新数据写入等操作,但这些操作不会对原有结果集构成影响。


如何实现呢?


可以创建一个时间点 Point In Time(PIT)保障搜索过程中保留特定事件点的索引状态。


Point In Time(PIT)是 Elasticsearch 7.10 版本之后才有的新特性。


PIT的本质:存储索引数据状态的轻量级视图。


如下示例能很好的解读 PIT 视图的内涵。


# 创建 PIT

POST kibana_sample_data_logs/_pit?keep_alive=1m

# 获取数据量 14074

POST kibana_sample_data_logs/_count

# 新增一条数据

POST kibana_sample_data_logs/_doc/14075

{

 "test":"just testing"

}

# 数据总量为 14075

POST kibana_sample_data_logs/_count

# 查询PIT,数据依然是14074,说明走的是之前时间点的视图的统计。

POST /_search

{

 "track_total_hits": true,

 "query": {

   "match_all": {}

 },

  "pit": {

   "id": "48myAwEXa2liYW5hX3NhbXBsZV9kYXRhX2xvZ3MWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAWdG1TOWFMTF9UdTZHdVZDYmhoWUljZwAAAAAAAAEN3RZGOFJCMGVrZVNndTk3U1I0SG81V3R3AAEWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAA"

 }

}

有了 PIT,search_after 的后续查询都是基于 PIT 视图进行,能有效保障数据的一致性。


search_after 分页查询可以简单概括为如下几个步骤。


步骤 1:创建 PIT 视图,这是前置条件不能省。


# Step 1: 创建 PIT

POST kibana_sample_data_logs/_pit?keep_alive=5m

返回结果如下:


{

 "id" : "48myAwEXa2liYW5hX3NhbXBsZV9kYXRhX2xvZ3MWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAWdG1TOWFMTF9UdTZHdVZDYmhoWUljZwAAAAAAAAEg5RZGOFJCMGVrZVNndTk3U1I0SG81V3R3AAEWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAA"

}

keep_alive=5m,类似scroll的参数,代表视图保留时间是 5 分钟,超过 5 分钟执行会报错如下:


 "type" : "search_context_missing_exception",

 "reason" : "No search context found for id [91600]"

步骤 2:创建基础查询语句,这里要设置翻页的条件。


# Step 2: 创建基础查询

GET /_search

{

 "size":10,

 "query": {

   "match" : {

     "host" : "elastic"

   }

 },

 "pit": {

    "id":  "48myAwEXa2liYW5hX3NhbXBsZV9kYXRhX2xvZ3MWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAWdG1TOWFMTF9UdTZHdVZDYmhoWUljZwAAAAAAAAEg5RZGOFJCMGVrZVNndTk3U1I0SG81V3R3AAEWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAA",

    "keep_alive": "1m"

 },

 "sort": [

   {"response.keyword": "asc"}

 ]

}

设置了PIT,检索时候就不需要再指定索引。


id 是基于步骤1 返回的 id 值。


排序 sort 指的是:按照哪个关键字排序。


在每个返回文档的最后,会有两个结果值,如下所示:


"sort" : [

         "200",

         4

       ]

其中,“200”就是我们指定的排序方式:基于 {"response.keyword": "asc"} 升序排列。


而 4 代表什么含义呢?


4 代表——隐含的排序值,是基于_shard_doc 的升序排序方式。


官方文档把这种隐含的字段叫做:tiebreaker (决胜字段),tiebreaker 等价于_shard_doc。


tiebreaker 本质含义:每个文档的唯一值,确保分页不会丢失或者分页结果数据出现重复(相同页重复或跨页重复)。


步骤3:实现后续翻页。


# step 3 : 开始翻页

GET /_search

{

 "size": 10,

 "query": {

   "match" : {

     "host" : "elastic"

   }

 },

 "pit": {

    "id":  "48myAwEXa2liYW5hX3NhbXBsZV9kYXRhX2xvZ3MWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAWdG1TOWFMTF9UdTZHdVZDYmhoWUljZwAAAAAAAAEg5RZGOFJCMGVrZVNndTk3U1I0SG81V3R3AAEWM2hGWXpxLXFSSGlfSmZIaXJWN0dxUQAA",

    "keep_alive": "1m"

 },

 "sort": [

   {"response.keyword": "asc"}

 ],

 "search_after": [                                

   "200",

   4

 ]

}

后续翻页都需要借助 search_after 指定前一页的最后一个文档的 sort 字段值。


如下代码所示:


 "search_after": [                                

   "200",

   4

 ]

显然,search_after 查询仅支持向后翻页。


2.2.2 search_after 查询优缺点及适用场景

search_after 优点


不严格受制于 max_result_window,可以无限制往后翻页。


ps:不严格含义:单次请求值不能超过 max_result_window;但总翻页结果集可以超过。


search_after 缺点


只支持向后翻页,不支持随机翻页。


search_after 适用场景


类似:今日头条分页搜索  https://m.toutiao.com/search


不支持随机翻页,更适合手机端应用的场景。

2.3 Scroll 遍历查询

2.3.1 Scroll 遍历查询定义与实战案例

相比于 From + size 和 search_after 返回一页数据,Scroll API 可用于从单个搜索请求中检索大量结果(甚至所有结果),其方式与传统数据库中游标(cursor)类似。


如果把  From + size 和 search_after 两种请求看做近实时的请求处理方式,那么 scroll 滚动遍历查询显然是非实时的。数据量大的时候,响应时间可能会比较长。


scroll 核心执行步骤如下:


步骤 1:指定检索语句同时设置 scroll 上下文保留时间。


实际上,scroll 已默认包含了 search_after 的PIT 的视图或快照功能。


从 Scroll 请求返回的结果反映了发出初始搜索请求时索引的状态,类似在那一个时刻做了快照。随后对文档的更改(写入、更新或删除)只会影响以后的搜索请求。


POST kibana_sample_data_logs/_search?scroll=3m

{

 "size": 100,

 "query": {

   "match": {

     "host": "elastic"

   }

 }

}

步骤 2:向后翻页继续获取数据,直到没有要返回的结果为止。


POST _search/scroll                                  

{

 "scroll" : "3m",

 "scroll_id":"FGluY2x1ZGVfY29udGV4dF91dWlkDXF1ZXJ5QW5kRmV0Y2gBFkY4UkIwZWtlU2d1OTdTUjRIbzVXdHcAAAAAAAGmkBZ0bVM5YUxMX1R1Nkd1VkNiaGhZSWNn"

}

scroll_id 值是步骤 1 返回的结果值。


2.3.2 Scroll 遍历查询优缺点及适用场景

   scroll 查询优点


支持全量遍历。


ps:单次遍历的 size 值也不能超过 max_result_window 大小。


   scroll 查询缺点


响应时间非实时。


保留上下文需要足够的堆内存空间。


scroll 查询适用场景


全量或数据量很大时遍历结果数据,而非分页查询。


官方文档强调:不再建议使用scroll API进行深度分页。如果要分页检索超过 Top 10,000+ 结果时,推荐使用:PIT + search_after。


3、小结

From+ size:需要随机跳转不同分页(类似主流搜索引擎)、Top 10000 条数据之内分页显示场景。


search_after:仅需要向后翻页的场景及超过Top 10000 数据需要分页场景。


Scroll:需要遍历全量数据场景 。


max_result_window:调大治标不治本,不建议调过大。


PIT:本质是视图。

image.png

本文说法有不严谨的地方,以官方文档为准。


欢迎大家就自己的分页实践进行留言讨论。


参考:


1. https://coralogix.com/log-analytics-blog/how-to-optimize-your-elasticsearch-queries-using-pagination


2. https://www.javatpoint.com/elasticsearch-pagination


3. https://www.elastic.co/guide/en/elasticsearch/reference/7.12/paginate-search-results.html

相关实践学习
使用阿里云Elasticsearch体验信息检索加速
通过创建登录阿里云Elasticsearch集群,使用DataWorks将MySQL数据同步至Elasticsearch,体验多条件检索效果,简单展示数据同步和信息检索加速的过程和操作。
ElasticSearch 入门精讲
ElasticSearch是一个开源的、基于Lucene的、分布式、高扩展、高实时的搜索与数据分析引擎。根据DB-Engines的排名显示,Elasticsearch是最受欢迎的企业搜索引擎,其次是Apache Solr(也是基于Lucene)。 ElasticSearch的实现原理主要分为以下几个步骤: 用户将数据提交到Elastic Search 数据库中 通过分词控制器去将对应的语句分词,将其权重和分词结果一并存入数据 当用户搜索数据时候,再根据权重将结果排名、打分 将返回结果呈现给用户 Elasticsearch可以用于搜索各种文档。它提供可扩展的搜索,具有接近实时的搜索,并支持多租户。
相关文章
|
存储 SQL 缓存
【ElasticSearch从入门到放弃系列 十】Elasticsearch深度分页查询方式讨论
【ElasticSearch从入门到放弃系列 十】Elasticsearch深度分页查询方式讨论
179 0
|
API 消息中间件 容器
避坑指南,Elasticsearch 分页查询的两个问题,你一定要知道
使用 Docker 部署完 RabbitMQ 服务,到 /var/log/rabbitmq 目录下一看,空空如也,并没有日志文件生成。 是没有日志吗?并非如此,日志都打在了标准输出上。
467 0
|
2月前
|
存储 安全 数据管理
如何在 Rocky Linux 8 上安装和配置 Elasticsearch
本文详细介绍了在 Rocky Linux 8 上安装和配置 Elasticsearch 的步骤,包括添加仓库、安装 Elasticsearch、配置文件修改、设置内存和文件描述符、启动和验证 Elasticsearch,以及常见问题的解决方法。通过这些步骤,你可以快速搭建起这个强大的分布式搜索和分析引擎。
68 5
|
3月前
|
存储 JSON Java
elasticsearch学习一:了解 ES,版本之间的对应。安装elasticsearch,kibana,head插件、elasticsearch-ik分词器。
这篇文章是关于Elasticsearch的学习指南,包括了解Elasticsearch、版本对应、安装运行Elasticsearch和Kibana、安装head插件和elasticsearch-ik分词器的步骤。
283 0
elasticsearch学习一:了解 ES,版本之间的对应。安装elasticsearch,kibana,head插件、elasticsearch-ik分词器。
|
4月前
|
NoSQL 关系型数据库 Redis
mall在linux环境下的部署(基于Docker容器),Docker安装mysql、redis、nginx、rabbitmq、elasticsearch、logstash、kibana、mongo
mall在linux环境下的部署(基于Docker容器),docker安装mysql、redis、nginx、rabbitmq、elasticsearch、logstash、kibana、mongodb、minio详细教程,拉取镜像、运行容器
mall在linux环境下的部署(基于Docker容器),Docker安装mysql、redis、nginx、rabbitmq、elasticsearch、logstash、kibana、mongo
|
5月前
|
数据可视化 Docker 容器
一文教会你如何通过Docker安装elasticsearch和kibana 【详细过程+图解】
这篇文章提供了通过Docker安装Elasticsearch和Kibana的详细过程和图解,包括下载镜像、创建和启动容器、处理可能遇到的启动失败情况(如权限不足和配置文件错误)、测试Elasticsearch和Kibana的连接,以及解决空间不足的问题。文章还特别指出了配置文件中空格的重要性以及环境变量中字母大小写的问题。
一文教会你如何通过Docker安装elasticsearch和kibana 【详细过程+图解】
|
5月前
|
JSON 自然语言处理 数据库
Elasticsearch从入门到项目部署 安装 分词器 索引库操作
这篇文章详细介绍了Elasticsearch的基本概念、倒排索引原理、安装部署、IK分词器的使用,以及如何在Elasticsearch中进行索引库的CRUD操作,旨在帮助读者从入门到项目部署全面掌握Elasticsearch的使用。
|
5月前
|
Ubuntu Oracle Java
如何在 Ubuntu VPS 上安装 Elasticsearch
如何在 Ubuntu VPS 上安装 Elasticsearch
65 0
|
5月前
|
存储 Ubuntu Oracle
在Ubuntu 14.04上安装和配置Elasticsearch的方法
在Ubuntu 14.04上安装和配置Elasticsearch的方法
53 0