大家好,我是不才陈某~
Elasticsearch 是一个实时的分布式搜寻与剖析引擎,在应用过程中,有一些典型的应用场景,比方分页、遍历等。
在应用关系型数据库中,咱们被告知要留神甚至被明确禁止应用深度分页,同理,在 Elasticsearch 中,也应该尽量避免应用深度分页。
这篇文章次要介绍 Elasticsearch 中分页相干内容!
关注公众号:码猿技术专栏,回复关键词:1111 获取阿里外部 Java 性能调优手册!
From/Size 参数
在 ES 中,分页查问默认返回最顶端的 10 条匹配 hits。
如果须要分页,须要应用 from 和 size 参数。
- from 参数定义了须要跳过的 hits 数,默认为 0;
- size 参数定义了须要返回的 hits 数目的最大值。
一个根本的 ES 查问语句是这样的:
POST /my_index/my_type/_search
{"query": { "match_all": {}},
"from": 100,
"size": 10
}
下面的查问示意从搜寻后果中取第 100 条开始的 10 条数据。
那么,这个查问语句在 ES 集群外部是怎么执行的呢?
在 ES 中,搜寻个别包含两个阶段,query 和 fetch 阶段,能够简略的了解,query 阶段确定要取哪些 doc,fetch 阶段取出具体的 doc。
Query 阶段
如上图所示,形容了一次搜寻申请的 query 阶段:·
- Client 发送一次搜寻申请,node1 接管到申请,而后,node1 创立一个大小为
from + size
的优先级队列用来存后果,咱们管 node1 叫 coordinating node。 - coordinating node 将申请播送到波及到的 shards,每个 shard 在外部执行搜寻申请,而后,将后果存到外部的大小同样为
from + size
的优先级队列里,能够把优先级队列了解为一个蕴含top N
后果的列表。 - 每个 shard 把暂存在本身优先级队列里的数据返回给 coordinating node,coordinating node 拿到各个 shards 返回的后果后对后果进行一次合并,产生一个全局的优先级队列,存到本身的优先级队列里。
在下面的例子中,coordinating node 拿到 (from + size) * 6
条数据,而后合并并排序后抉择后面的 from + size
条数据存到优先级队列,以便 fetch 阶段应用。
另外,各个分片返回给 coordinating node 的数据用于选出前 from + size
条数据,所以,只须要返回惟一标记 doc 的 _id
以及用于排序的 _score
即可,这样也能够保障返回的数据量足够小。
coordinating node 计算好本人的优先级队列后,query 阶段完结,进入 fetch 阶段。
Fetch 阶段
query 阶段晓得了要取哪些数据,然而并没有取具体的数据,这就是 fetch 阶段要做的。
上图展现了 fetch 过程:
- coordinating node 发送 GET 申请到相干 shards。
- shard 依据 doc 的
_id
取到数据详情,而后返回给 coordinating node。 - coordinating node 返回数据给 Client。
coordinating node 的优先级队列里有from + size
个_doc _id
,然而,在 fetch 阶段,并不需要取回所有数据,在下面的例子中,前 100 条数据是不须要取的,只须要取优先级队列里的第 101 到 110 条数据即可。
须要取的数据可能在不同分片,也可能在同一分片,coordinating node 应用 multi-get 来防止屡次去同一分片取数据,从而进步性能。
这种形式申请深度分页是有问题的:
咱们能够假如在一个有 5 个主分片的索引中搜寻。当咱们申请后果的第一页(后果从 1 到 10),每一个分片产生前 10 的后果,并且返回给 协调节点,协调节点对 50 个后果排序失去全副后果的前 10 个。
当初假如咱们申请第 1000 页—后果从 10001 到 10010。所有都以雷同的形式工作除了每个分片不得不产生前 10010 个后果以外。而后协调节点对全副 50050 个后果排序最初抛弃掉这些后果中的 50040 个后果。
对后果排序的老本随分页的深度成指数回升。
留神 1:
size 的大小不能超过 index.max_result_window
这个参数的设置,默认为 10000。
如果搜寻 size 大于 10000,须要设置 index.max_result_window
参数
PUT _settings
{
"index": {"max_result_window": "10000000"}
}
留神 2:
_doc
将在将来的版本移除,详见:
- https://www.elastic.co/cn/blog/moving-from-types-to-typeless-…
- https://elasticsearch.cn/article/158
深度分页问题
Elasticsearch 的 From/Size 形式提供了分页的性能,同时,也有相应的限度。
举个例子,一个索引,有 10 亿数据,分 10 个 shards,而后,一个搜寻申请,from=1000000,size=100,这时候,会带来重大的性能问题:CPU,内存,IO,网络带宽。
在 query 阶段,每个 shards 须要返回 1000100 条数据给 coordinating node,而 coordinating node 须要接管10 * 1000
,100 条数据,即便每条数据只有 _doc _id
和 _score
,这数据量也很大了?
在另一方面,咱们意识到,这种深度分页的申请并不合理,因为咱们是很少人为的看很前面的申请的,在很多的业务场景中,都间接限度分页,比方只能看前 100 页。
比方,有 1 千万粉丝的微信大 V,要给所有粉丝群发音讯,或者给某省粉丝群发,这时候就须要获得所有符合条件的粉丝,而最容易想到的就是利用 from + size 来实现,不过,这个是不事实的,这时,能够采纳 Elasticsearch 提供的其余形式来实现遍历。
深度分页问题大抵能够分为两类:
- 随机深度分页:随机跳转页面
- 滚动深度分页:只能一页一页往下查问
上面介绍几个官网提供的深度分页办法
Scroll
Scroll 遍历数据
咱们能够把 scroll 了解为关系型数据库里的 cursor,因而,scroll 并不适宜用来做实时搜寻,而更适宜用于后盾批处理工作,比方群发。
这个分页的用法,不是为了实时查问数据 ,而是为了 一次性查问大量的数据(甚至是全副的数据)。
因为这个 scroll 相当于保护了一份以后索引段的快照信息,这个快照信息是你执行这个 scroll 查问时的快照。在这个查问后的任何新索引进来的数据,都不会在这个快照中查问到。
然而它绝对于 from 和 size,不是查问所有数据而后剔除不要的局部,而是记录一个读取的地位,保障下一次疾速持续读取。
不思考排序的时候,能够联合 SearchType.SCAN
应用。
scroll 能够分为初始化和遍历两部,初始化时将 所有合乎搜寻条件的搜寻后果缓存起来(留神,这里只是缓存的 doc_id,而并不是真的缓存了所有的文档数据,取数据是在 fetch 阶段实现的),能够设想成快照。
在遍历时,从这个快照里取数据,也就是说,在初始化后,对索引插入、删除、更新数据都不会影响遍历后果。
根本应用
POST /twitter/tweet/_search?scroll=1m
{
"size": 100,
"query": {
"match" : {"title" : "elasticsearch"}
}
}
初始化指明 index 和 type,而后,加上参数 scroll,示意暂存搜寻后果的工夫,其它就像一个一般的 search 申请一样。
会返回一个 _scroll_id
,_scroll_id
用来下次取数据用。
遍历
POST /_search?scroll=1m
{"scroll_id":"XXXXXXXXXXXXXXXXXXXXXXX I am scroll id XXXXXXXXXXXXXXX"}
这里的 scroll_id
即 上一次遍历取回的 _scroll_id
或者是初始化返回的_scroll_id
,同样的,须要带 scroll 参数。
反复这一步骤,直到返回的数据为空,即遍历实现。
留神,每次都要传参数 scroll,刷新搜寻后果的缓存工夫 。另外, 不须要指定 index 和 type。
设置 scroll 的时候,须要使搜寻后果缓存到下一次遍历实现,同时,也不能太长,毕竟空间无限。
优缺点
毛病:
- scroll_id 会占用大量的资源(特地是排序的申请)
- 同样的,scroll 后接超时工夫,频繁的发动 scroll 申请,会呈现一些列问题。
- 是生成的历史快照,对于数据的变更不会反映到快照上。
长处:
实用于非实时处理大量数据的状况,比方要进行数据迁徙或者索引变更之类的。
Scroll Scan
ES 提供了 scroll scan 形式进一步提高遍历性能,然而 scroll scan 不反对排序,因而 scroll scan 适宜不须要排序的场景
根本应用
Scroll Scan 的遍历与一般 Scroll 一样,初始化存在一点差异。
POST /my_index/my_type/_search?search_type=scan&scroll=1m&size=50
{"query": { "match_all": {}}
}
须要指明参数:
search_type
:赋值为 scan,示意采纳 Scroll Scan 的形式遍历,同时通知 Elasticsearch 搜寻后果不须要排序。- scroll:同上,传工夫。
- size:与一般的 size 不同,这个 size 示意的是每个 shard 返回的 size 数,最终后果最大为
number_of_shards * size
。
Scroll Scan 与 Scroll 的区别
- Scroll-Scan 后果 没有排序,按 index 程序返回,没有排序,能够进步取数据性能。
- 初始化时只返回
_scroll_id
,没有具体的 hits 后果 - size 管制的是每个分片的返回的数据量,而不是整个申请返回的数据量。
Sliced Scroll
如果你数据量很大,用 Scroll 遍历数据那的确是承受不了,当初 Scroll 接口能够并发来进行数据遍历了。
每个 Scroll 申请,能够分成多个 Slice 申请,能够了解为切片,各 Slice 独立并行,比用 Scroll 遍历要快很多倍。
POST /index/type/_search?scroll=1m
{"query": { "match_all": {}},
"slice": {
"id": 0,
"max": 5
}
}
POST ip:port/index/type/_search?scroll=1m
{"query": { "match_all": {}},
"slice": {
"id": 1,
"max": 5
}
}
上边的示例能够独自申请两块数据,最终五块数据合并的后果与间接 scroll scan 雷同。
其中 max 是分块数,id 是第几块。
官网文档中倡议 max 的值不要超过 shard 的数量,否则可能会导致内存爆炸。
Search After
Search_after
是 ES 5 新引入的一种分页查问机制,其原理简直就是和 scroll 一样,因而代码也简直是一样的。
根本应用:
第一步:
POST twitter/_search
{
"size": 10,
"query": {
"match" : {"title" : "es"}
},
"sort": [{"date": "asc"},
{"_id": "desc"}
]
}
返回出的后果信息:
{
"took" : 29,
"timed_out" : false,
"_shards" : {
"total" : 1,
"successful" : 1,
"skipped" : 0,
"failed" : 0
},
"hits" : {
"total" : {
"value" : 5,
"relation" : "eq"
},
"max_score" : null,
"hits" : [
{...},
"sort" : [...]
},
{...},
"sort" : [
124648691,
"624812"
]
}
]
}
}
下面的申请会为每一个文档返回一个蕴含 sort 排序值的数组。
这些 sort 排序值能够被用于 search_after
参数里以便抓取下一页的数据。
比方,咱们能够应用最初的一个文档的 sort 排序值,将它传递给 search_after
参数:
GET twitter/_search
{
"size": 10,
"query": {
"match" : {"title" : "es"}
},
"search_after": [124648691, "624812"],
"sort": [{"date": "asc"},
{"_id": "desc"}
]
}
若咱们想接着上次读取的后果进行读取下一页数据,第二次查问在第一次查问时的语句根底上增加search_after
,并指明从哪个数据后开始读取。
基本原理
es 保护一个实时游标,它以上一次查问的最初一条记录为游标,不便对下一页的查问,它是一个无状态的查问,因而每次查问的都是最新的数据。
因为它采纳记录作为游标,因而SearchAfter 要求 doc 中至多有一条全局惟一变量(每个文档具备一个惟一值的字段应该用作排序标准)
优缺点
长处:
- 无状态查问,能够避免在查问过程中,数据的变更无奈及时反映到查问中。
- 不须要保护
scroll_id
,不须要保护快照,因而能够防止耗费大量的资源。
毛病:
- 因为无状态查问,因而在查问期间的变更可能会导致跨页面的不一值。
- 排序程序可能会在执行期间发生变化,具体取决于索引的更新和删除。
- 至多须要制订一个惟一的不反复字段来排序。
- 它不适用于大幅度跳页查问,或者全量导出,对第 N 页的跳转查问相当于对 es 一直反复的执行 N 次 search after,而全量导出则是在短时间内执行大量的反复查问。
SEARCH_AFTER
不是自在跳转到任意页面的解决方案,而是并行滚动多个查问的解决方案。
总结
分页形式 | 性能 | 长处 | 毛病 | 场景 |
---|---|---|---|---|
from + size | 低 | 灵活性好,实现简略 | 深度分页问题 | 数据量比拟小,能容忍深度分页问题 |
scroll | 中 | 解决了深度分页问题 | 无奈反馈数据的实时性(快照版本)保护老本高,须要保护一个 scroll_id | 海量数据的导出须要查问海量后果集的数据 |
search_after | 高 | 性能最好不存在深度分页问题可能反映数据的实时变更 | 实现简单,须要有一个全局惟一的字段间断分页的实现会比较复杂,因为每一次查问都须要上次查问的后果,它不适用于大幅度跳页查问 | 海量数据的分页 |
ES7 版本变更
参照:https://www.elastic.co/guide/en/elasticsearch/reference/maste…
在 7.*
版本中,ES 官网不再举荐应用 Scroll 办法来进行深分页,而是举荐应用带 PIT 的 search_after
来进行查问;
从 7.*
版本开始,您能够应用 SEARCH_AFTER
参数通过上一页中的一组排序值检索下一页命中。
应用 SEARCH_AFTER
须要多个具备雷同查问和排序值的搜寻申请。
如果这些申请之间产生刷新,则后果的程序可能会更改,从而导致页面之间的后果不统一。
为防止出现这种状况,您能够创立一个工夫点 (PIT) 来在搜寻过程中保留以后索引状态。
POST /my-index-000001/_pit?keep_alive=1m
返回一个 PIT ID:{"id": "46ToAwMDaWR5BXV1aWQyKwZub2RlXzMAAAAAAAAAACoBYwADaWR4BXV1aWQxAgZub2RlXzEAAAAAAAAAAAEBYQADaWR5BXV1aWQyKgZub2RlXzIAAAAAAAAAAAwBYgACBXV1aWQyAAAFdXVpZDEAAQltYXRjaF9hbGw_gAAAAA=="}
在搜寻申请中指定 PIT:
GET /_search
{
"size": 10000,
"query": {
"match" : {"user.id" : "elkbee"}
},
"pit": {
"id": "46ToAwMDaWR5BXV1aWQyKwZub2RlXzMAAAAAAAAAACoBYwADaWR4BXV1aWQxAgZub2RlXzEAAAAAAAAAAAEBYQADaWR5BXV1aWQyKgZub2RlXzIAAAAAAAAAAAwBYgACBXV1aWQyAAAFdXVpZDEAAQltYXRjaF9hbGw_gAAAAA==",
"keep_alive": "1m"
},
"sort": [{"@timestamp": {"order": "asc", "format": "strict_date_optional_time_nanos", "numeric_type" : "date_nanos"}}
]
}
性能比照
别离分页获取 1 - 10
,49000 - 49010
,99000 - 99010
范畴各 10 条数据(前提 10w 条),性能大抵是这样:
向前翻页
对于向前翻页,ES 中没有相应 API,然而依据官网说法(https://github.com/elastic/elasticsearch/issues/29449),ES 中的向前翻页问题能够通过翻转排序形式来实现即:
- 对于某一页,正序
search_after
该页的最初一条数据 id 为下一页,则逆序search_after
该页的第一条数据 id 则为上一页。 - 国内论坛上,有人应用缓存来解决上一页的问题:https://elasticsearch.cn/question/7711
总结
- 如果数据量小(from+size 在 10000 条内),或者只关注后果集的 TopN 数据,能够应用 from/size 分页,简略粗犷
- 数据量大,深度翻页,后盾批处理工作(数据迁徙)之类的工作,应用 scroll 形式
- 数据量大,深度翻页,用户实时、高并发查问需要,应用 search after 形式
集体思考
Scroll 和 search_after
原理基本相同,他们都采纳了游标的形式来进行深分页。
这种形式尽管可能肯定水平上解决深分页问题。然而,它们并不是深分页问题的终极解决方案,深分页问题 必须防止!!。
对于 Scroll,无可避免的要保护 scroll_id
和历史快照,并且,还必须保障 scroll_id
的存活工夫,这对服务器是一个微小的负荷。
对于Search_After
,如果容许用户大幅度跳转页面,会导致短时间内频繁的搜寻动作,这样的效率十分低下,这也会减少服务器的负荷,同时,在查问过程中,索引的增删改会导致查问数据不统一或者排序变动,造成后果不精确。
Search_After
自身就是一种业务折中计划,它不容许指定跳转到页面,而只提供下一页的性能。
Scroll 默认你会在后续将所有符合条件的数据都取出来,所以,它只是搜寻到了所有的符合条件的 doc_id
(这也是为什么官网举荐用doc_id
进行排序,因为自身缓存的就是doc_id
,如果用其余字段排序会减少查问量),并将它们排序后保留在协调节点(coordinate node),然而并没有将所有数据进行 fetch,而是每次 scroll,读取 size 个文档,并返回此次读取的最初一个文档以及上下文状态,用以告知下一次须要从哪个 shard 的哪个文档之后开始读取。
这也是为什么官网不举荐 scroll 用来给用户进行实时的分页查问,而是适宜于大批量的拉取数据,因为它从设计上就不是为了实时读取数据而设计的。