Node.js精进(12)——ElasticSearch
ElasticSearch(简称 ES)是一款基于 Lucene 的分布式、可扩展、RESTful 风格的全文检索和数据分析引擎,擅长实时处理 PB 级别的数据。
一、基本概念
1)Lucene
Lucene 是一款开源免费、成熟权威、高性能的全文检索库,是 ES 实现全文检索的核心基础,而检索的关键正是倒排索引。
2)倒排索引
索引的目的是加快查询速度,尽快查出符合条件的数据。
正排索引就像翻书一样,先查目录,然后锁定页码,再去看内容。而倒排索引正好与其相反,通过对内容的分词,建立内容到文档 ID 之间的映射关系,如下图所示(来源于elasticsearch原理及入门)。
倒排索引包括两部分: Term Dictionary(单词词典)和 Posting List(倒排列表)。
Term Dictionary 记录了文档单词,以及单词和倒排列表的关系。Posting List 则是记录了 Term 在文档中的位置以及其他信息,主要包括文档 ID、词频(Term 在文档中出现的次数,用来计算相关性评分),位置以及偏移(实现搜索高亮)。
3)压缩算法
为了搜索能高性能,需要将倒排列表放入内存中,但是海量的文档必然会增加表的尺寸,为了节约空间,Lucene 使用了两种压缩算法:FOR(Frame Of Reference)和 RBM(RoaringBitmap)。
FOR 算法的原理就是通过增量,将原来的大数变成小数,仅存储增量值,最后通过字节存储,具体分为 3 步:
- 将排序的整数列表转换成 Delta 列表,第二排的 227 是增量值(300 - 73),其余值依次计算。
- 切分成 blocks,每个 block 是 256 个 Delta,这里为了简化一下,搞成 3 个 Delta。
- 看下每个 block 最大的 Delta 是多少。下图的第一个,最大是 227,最接近的 2 次幂是 256(8bits),于是规定这个 block 里都用 8bits 来编码(绿色的 header 就是 8);第二个最大的是 30,最接近的 2 次幂是 32(5bits),于是规定这个 block 里都用 5bits 来编码。
FOR 压缩算法适用于间隔比较小稠密的文档 ID 列表,如1、2、3、5、8.......。假如遇到间隔较大稀疏的文档 ID 列表,如 1000、62101、131385、132052、191173、196658,就更适合通过 RBM 算法来压缩。
RBM 算法的核心就是把数据表示成 32 位的二进制,分为高 16 和低 16 进行分别存储,最大值就是 2 的 16 次方(即 65536)。下图描述了具体的压缩步骤(来源于elasticsearch原理及入门):
- 每个数字除以 65536 会得到一个商和余数。
- 用(商,余数)的组合表示每一组 ID,范围都在 0 ~ 65535 之内。
- 其中商为该数字(以 196658 为例)的二进制的前 16 位,余数为该数字的二进制的后 16 位。
- 再将商提取出来作为 short key,将关联的余数整合在一起,例如商是 0,则 1000 和 62101 重新组合。
4)FST
在数据写入的时候,Lucene 会为原始数据中的每个 Term 生成对应的倒排索引,这就会让倒排索引的数据量变得很大。而倒排索引对应的倒排列表文件又是存储在硬盘上的,如果每次查询都直接去磁盘中读取,那就会严重影响全文检索的效率。
因此需要一种方式可以快速定位到倒排索引中的 Term,Lucene 使用了 FST(Finite State Transducer)有限状态转换器来实现二级索引的设计,这是一种类似 Trie 树的算法。
Trie 树是一种树形结构,哈希树的变种,经常被搜索引擎系统用于文本词频统计。可利用字符串的公共前缀来减少查询时间,最大限度地减少无谓的字符串比较,查询效率比哈希树高。它有 3 个基本性质:
- 根节点不包含字符,除根节点外每一个节点都只包含一个字符。
- 从根节点到某一节点,路径上经过的字符连接起来,为该节点对应的字符串。
- 每个节点的所有子节点包含的字符都不相同。
假设有两个 Term:school 和 cool,它们后面的字符一致,可以通过将原先的 Trie 树中的后缀字符进行合并来进一步的压缩空间。优化后的 trie 树就是 FST,如下图所示(来源于Elasticsearch核心概念):
5)术语
ES 是分布式数据库,允许多台服务器协同工作,每台服务器可以运行多个实例。单个实例称为一个节点(node),一组节点构成一个集群(cluster)。
在上图中,包含三类节点:
- 主节点(Master Node),为确保一个集群的稳定,分离主节点和数据节点,主要职责是和集群操作相关的内容,如创建或删除索引,跟踪哪些节点是集群的一部分,并决定哪些分片分配给相关的节点。
- 数据节点(Data Node),存储索引数据的节点,主要对文档进行增删改查、聚合等操作。
- 协调节点(Coordinator Node),该节点只处理路由请求、分发索引等操作,相当于一个智能负载平衡器,协调节点将请求转发给存储数据的 Data Node。每个 Data Node 会将结果返回协调节点,协调节点收集完数据后,将每个 Data Node 的结果合并为单个全局结果。
分片(shared)是底层的工作单元,文档(document)保存在分片内,分片又被分配到集群内的各个节点里,每个分片仅保存全部数据的一部分。注意,分片不是随意进行设定的,而是需要根据实际的生产环境提前进行数据存储的容量规划,若设置的过大或过小都会影响 ES 集群的整体性能。
索引(index)是一类文档的集合,而文档是具体的一条数据,注意,从 ElasticSearch 8 开始,彻底移除了 Type 的概念。
为了便于理解,相关概念与关系型数据库(MySQL)的对比如下:
MySQL | ElasticSearch |
Table | Index |
Row | Doucment |
Column | Field |
Schema | Mapping |
SQL | DSL |
二、实战应用
1)安装
在官网可以下载各种操作系统版本的 ES,当进入下载页面时会自动切换成当前电脑的系统。
下载完成后,就可以执行第二步,运行 bin 目录中 elasticsearch 可执行文件,简单点就是将其拖到命令行窗口中。
在安装成功后,保存给出的密码和 token。
2)Kibana
官方提供了一套可视化操作 ES 的系统:Kibana,在下载完成后,运行 bin 目录中的 kibana 文件。
耐心等待,安装成功后,在命令窗口会给出一条地址。
在初始化时会要求填入之前保存的 token,点击 Configure 按钮,若弹出验证码,则将上图中的 code 参数复制过来,配置完成后进入登录页面。
在登录时会用到默认账号 elastic,上一节保存的密码,点击确定进入主页,在左侧菜单中找到 Dev Tools。
点击后就能进入可运行 ES RESTful API 的操作界面。
若 Kibana 启动不了,报错如下:
Kibana server is not ready yet.
此时可以打开 config/kibana.yml 中的配置文件,翻到最后,很可能是 hosts 中的 IP 地址有问题,因为电脑重新联网时,IP 地址很有可能变换了,将其改成 localhost 问题就能迎刃而解。
elasticsearch.hosts: ['https://172.21.10.10:9200']
elasticsearch.serviceAccountToken: AAEAAWVsYXN0aWMva2liYW5
elasticsearch.ssl.certificateAuthorities: [/Users/pwstrick/code/kibana/data/ca_1699243503862.crt]
xpack.fleet.outputs: [{id: fleet-default-output, name: default, is_default: true,
is_default_monitoring: true, type: elasticsearch, hosts: ['https://172.21.10.10:9200'],
ca_trusted_fingerprint: 1b6c0b97e18f22efdd4925a95a4a0dc898de5072e3d6c45938b8d2f0a7f920fb}]
3)RESTful API
ES 提供了对 Document 进行增删改查的常规接口,例如使用 Bulk 接口插入一条数据,_index 就相当于数据库表,第三行就是具体的字段名称和值。
POST _bulk
{"index": {"_id": 862024079,"_index": "web_monitor_2023.11"}}
{"id":862024079,"project":"game","project_subdir":"chat","category":"ajax",
"message":"{\"type\":\"GET\",\"url\":\"https://static.xxx.me/xxx.json\",\"status\":200,\"endBytes\":\"80.43KB\",\"interval\":\"9ms\"}",
"key":"80c89d32b27f8f7d43fa8470aeba3f3a","source":"","identity":"xe990bhs4j","referer":"https://www.xxx.me/chat.html",
"message_type":"get","message_status":200,"message_path":"xxx.json","day":"20231103","hour":15,"minute":29,"ctime":1698996585,
"ip":"0.0.0.0","os_name":"iOS","os_version":"15.4.1","app_version":"5.36.1","author":"张三",
"fingerprint":"38eab40b373220bea1bab2933649c","country":"中国","province":"广东省","city":"佛山市","isp":"电信","digit":1}
若要更新或删除一条记录,也可以在 Bulk 接口完成,格式参考如下,更新语句需要包含待更新的数据。
POST _bulk
{ "delete" : {"_id" : "2", "_index" : "web_monitor_2023.11" } } { "update" : {"_id" : "1", "_index" : "web_monitor_2023.11"} }
{ "doc" : {"field" : "value"} }
使用 Search 接口做查询,格式参考 GET /<target>/_search,其中 target 可以理解为 Index(相当于数据库表的名称)。
GET web_monitor_2023.11/_search
响应的 JSON 结构字段包含众多(如下所示),took 是搜索耗费的毫秒数;_shards 中的 total 代表本次搜索一共使用的分片数量;hits 中的 total 代表本次搜索得到的结果数,默认最大值为 1W,max_score 指搜索结果中相关度得分的最大值,默认搜索结果会按照相关度得分降序排列,hits 就是命中的数据列表,而其中的 _score 是单个文档的相关度得分,_source 就是原始数据的 JSON 内容。
{
"took": 6, // 搜索耗费的毫秒数
"timed_out": false,
"_shards": {
"total": 1, // 本次搜索一共使用的分片数量
"successful": 1,
"skipped": 0,
"failed": 0
},
"hits": {
"total": {
"value": 1, // 本次搜索得到的结果数,默认最大值为 1W
"relation": "eq"
},
"max_score": 1, // 搜索结果中相关度得分的最大值
"hits": [
{
"_index": "web_monitor_2023.11",
"_id": "862024079",
"_score": 1, // 单个文档的相关度得分
"_source": { // 原始数据的 JSON 内容
"id": 862024079,
"project": "game",
"project_subdir": "chat",
"category": "ajax",
"fingerprint": "38eab40b373220bea1baee7b2933649c",
"country": "中国",
"province": "广东省",
"city": "佛山市",
"isp": "电信",
"digit": 1
}
}
]
}
}
如果要计算搜索结果真实的数据量,可以参考 Count 接口,格式为 GET /<target>/_count。
4)索引模板
索引模板(Index Template)允许用户在创建索引时,引用已保存的模板来减少配置项,在 MySQL 中就相当于创建表结构。
Elasticsearch 的索引模板功能以 7.8 版本为界,两个版本的主要区别是模板之间复用方式。
- 老版本:使用优先级(order)关键字实现,当创建索引匹配到多个索引模板时,高优先级会继承并覆盖低优先级的模板配置,最终多个模板共同起作用。
- 新版本:删除了 order 关键字,引入了组件模板(Component Template)的概念。在声明索引模板时可以引用多个组件模板,当创建索引匹配到多个索引模板时,选最高权重的那个。
老版本会造成用户在创建索引时,不能明确知道自己到底用了多少模板,索引配置在继承覆盖的过程中容易出错。
创建或更新一个老版索引模板,需要向 /_template 发送 PUT 请求,配置包括 aliases、settings、mappings、order 等字段。
PUT _template/web_monitor
{
order: 0,
index_patterns: ["web_monitor_*"],
settings: {
index: {
number_of_shards: 1
}
},
mappings: {
dynamic: "strict",
properties: {
app_version: {
type: "keyword"
},
ctime: {
format: "strict_date_optional_time||epoch_second",
type: "date"
},
digit: {
type: "keyword",
fields: {
num: {
type: "integer"
}
}
},
author: {
type: "keyword"
},
ip: {
type: "ip"
}
}
},
aliases: {
web_monitor: {}
}
}
新版本索引自动配置功能,需要通过组件模板和索引模板来完成。
在组件模板中可配置的字段包括:aliases、settings 和 mappings,组件模板只有在被索引模板引用时,才会发挥作用。当需要创建或更新一个组件模板时,向 /_component_template 发送 PUT 请求即可。
PUT /_component_template/ct1
{
"template": {
"settings": {
"index.number_of_shards": 2
}
}
}
PUT /_component_template/ct2
{
"template": {
"settings": {
"index.number_of_replicas": 0
},
"mappings": {
"properties": {
"@timestamp": {
"type": "date"
}
}
}
}
}
创建或更新一个索引模板的方式都是向 /_index_template 发送 1 个 POST 请求。
POST /_index_template/_simulate
{
"index_patterns": ["my*"],
"template": {
"settings" : {
"index.number_of_shards" : 3
}
},
"composed_of": ["ct1", "ct2"]
}
5)搜索
下面是一组查询条件,query、from、size 和 sort 平级,分别表示查询条件、页码、页数和排序规则。
{
query: {
bool: { // 布尔查询
must: [
[
{
multi_match: {
query: "精确",
fields: ["message", "title"],
type: "best_fields"
}
}
]
],
filter: [
{
term: {
category: "error"
}
},
{
term: {
project: "backend-app"
}
},
{
term: {
message_type: "runtime"
}
},
{
range: {
ctime: {
gte: 1699286400,
lt: 1699372800
}
}
}
]
}
},
from: 0,
size: 10,
sort: [
{
id: {
order: "DESC"
}
}
]
}
布尔查询(bool),只有符合整个布尔条件的文档才会被搜索出来,支持 4 种组合类型:
- must:可包含多个查询条件,每个条件都被满足才能命中,每次查询需要计算相关度得分。
- should:可包含多个查询条件,只要满足一个条件就能命中,匹配到结果越多,相关度得分也越高。
- filter:与 must 作用类似,但是不计算相关度得分,结果在一定条件下会被缓存。
- must_not:与 must 作用相反,并且也不计算相关度得分,结果在一定条件下会被缓存。
多字段匹配(multi_match)允许用同一段文本检索多个字段,其中 best_fields 是默认的搜索方式,搜索文本与哪个字段相关度最高,就使用最佳字段中的 _score。
ES 内置了 8 种文本分析器,但对于中文的支持并不友好,无法准确的反映中文文本的语义,所以对于中文需要安装另一款分析器:ik。
除了常规的全文检索和精准查询之外,ES 还支持经纬度搜索,包括圆形、矩形和多边形范围内的搜索。
6)聚合
当需要对数据做分析时,就需要对数据进行聚合。在 MySQL 中常用的就是 sum()、group by 等语法。
ES 提供的聚合分为 3 大类:
- 度量聚合:计算搜索结果在某个字段上的数量统计指标,包括平均值、最大值、最小值、求和、基数(唯一值)、百分比、头部命中等。
- 桶聚合:在某个字段上划定一些区间,每个区间是一个桶,统计结果能明确每个桶中的文档数量。桶聚合还能嵌套其他的桶聚合或度量聚合来进行更为复杂的指标计算,例如词条、直方图、缺失等聚合。
- 管道聚合:把桶聚合统计的结果作为输入来继续做聚合统计,在结果中追加一些额外的统计数据。
下面是一个桶聚合的例子,在查询条件中使用了 ES 特有的时间范围语法糖(now-7d/d)。
聚合部分要使用 aggs 属性包裹,其子属性 date 自定义的聚合名称(在搜索结果中也会包含这个自定义的名称),date_histogram 是聚合类型,以天为间隔,计算每天符合条件的数量。
{
query: {
bool: {
filter: [
{
term: {
category: "error"
}
},
{
range: {
ctime: {
gt: "now-7d/d", // 当前时间减去 7 天
lte: "now/d"
}
}
}
]
}
},
aggs: {
date: {
date_histogram: {
field: "ctime", // 字段名称
interval: "day", // 以天为间隔
time_zone: "+08:00"
}
}
}
}
聚合结果与查询结果类似,也会包含符合查询条件的文档列表,但是还会多一个 aggregations 属性。
其 date 属性就是之前自定义的聚合名称,buckets 中就是聚合结果,key 是聚合的字段值,doc_count 是计算的结果值,key_as_string 是格式化后的日期值,可在查询时指定格式。
{
took: 245,
timed_out: false,
_shards: {
total: 2,
successful: 2,
skipped: 0,
failed: 0
},
hits: {
total: {
value: 3799,
relation: "eq"
},
max_score: 0,
hits: [{}, {}]
},
aggregations: {
date: {
buckets: [
{
key_as_string: "2023-11-02T00:00:00.000+08:00",
key: 1698854400000,
doc_count: 451
},
{
key_as_string: "2023-11-03T00:00:00.000+08:00",
key: 1698940800000,
doc_count: 594
},
{
key_as_string: "2023-11-04T00:00:00.000+08:00",
key: 1699027200000,
doc_count: 612
}
]
}
}
}
参考资料:
Frame of Reference 和 Roaring Bitmaps
elasticsearch-Index template 索引模板
Node.js精进(12)——ElasticSearch的更多相关文章
- Node.js V0.12 新特性之性能优化
v0.12悠长的开发周期(已经过去九个月了,并且还在继续,是有史以来最长的一次)让核心团队和贡献者们有充分的机会对性能做一些优化. 本文会介绍其中最值得注意的几个. http://www.infoq. ...
- 【译】 Node.js v0.12的新特性 -- 性能优化
原文: https://strongloop.com/strongblog/performance-node-js-v-0-12-whats-new/ January 21, 2014/in Comm ...
- 【译】 Node.js v0.12的新特性 -- Cluster模式采用Round-Robin负载均衡
原文:https://strongloop.com/strongblog/whats-new-in-node-js-v0-12-cluster-round-robin-load-balancing 本 ...
- Node.js V0.12新特性之性能优化
v0.12悠长的开发周期(已经过去九个月了,并且还在继续,是有史以来最长的一次)让核心团队和贡献者们有充分的机会对性能做一些优化.本文会介绍其中最值得注意的几个. 支持塞住模式的可写流 现在可写流可以 ...
- Node.js 0.12: 正确发送HTTP POST请求
Node.js 0.12: 正确发送HTTP POST请求 本文针对版本:Node.js 0.12.4 之前写过一篇Node.js发送和接收HTTP的GET请求的文章,今天再写一篇,讲发送POST的请 ...
- Node.js精进(8)——错误处理
在 Node.js 中,提供了 error 模块,并且内置了标准的 JavaScript 错误,常见的有: EvalError:在调用 eval() 函数时出现问题时抛出该错误. SyntaxErro ...
- Node.js学习(12)----Web应用开发
1.使用http模块 Node.js 由于不需要另外的 HTTP 服务器,因此减少了一层抽象,给性能带来不少提升, 但同时也因此而提高了开发难度.举例来说,我们要实现一个 POST 数据的表单,例如: ...
- Node.js精进(1)——模块化
模块化是一种将软件功能抽离成独立.可交互的软件设计技术,能促进大型应用程序和系统的构建. Node.js内置了两种模块系统,分别是默认的CommonJS模块和浏览器所支持的ECMAScript模块. ...
- Node.js精进(2)——异步编程
虽然 Node.js 是单线程的,但是在融合了libuv后,使其有能力非常简单地就构建出高性能和可扩展的网络应用程序. 下图是 Node.js 的简单架构图,基于 V8 和 libuv,其中 Node ...
- Node.js精进(3)——流
在 JavaScript 中,一般只处理字符串层面的数据,但是在 Node.js 中,需要处理网络.文件等二进制数据. 由此,引入了Buffer和Stream的概念,两者都是字节层面的操作. Buff ...
随机推荐
- 青少年CTF平台-Web-Robots
题目信息 题目名称:Robots 题目描述:昨天十三年社团讲课,讲了Robots.txt的作用,小刚上课没有认真听课正在着急,你能不能帮帮忙? 题目难度:一颗星 解题过程 访问题目链接 在这里插入图片 ...
- 服务端apk打包教程
本文我将给大家介绍一个 apk 打包工具 VasDolly 的使用介绍.原理以及如何在服务端接入 VasDolly 进行服务端打渠道包操作. 使用介绍 VasDolly 是一个快速多渠道打包工具,同时 ...
- .NET技术:懒惰与沉淀的平衡之道
在过去的很多年里,我一直默默搬砖,而我们聚在博客园,目的只有一个:沉淀并为更多的.NET开发者提供更好的帮助. 疫情3年,个人经历了太多事情,感觉懒惰是最大的敌人.然而,在这里,我收获了许多宝贵的经验 ...
- quarkus依赖注入之八:装饰器(Decorator)
欢迎访问我的GitHub 这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos 本篇概览 本篇是<quarkus依赖注入> ...
- 2.0 Python 数据结构与类型
数据类型是编程语言中的一个重要概念,它定义了数据的类型和提供了特定的操作和方法.在 python 中,数据类型的作用是将不同类型的数据进行分类和定义,例如数字.字符串.列表.元组.集合.字典等.这些数 ...
- 《Java极简设计模式》第01章:单例模式(Singleton)
作者:冰河 星球:http://m6z.cn/6aeFbs 博客:https://binghe.gitcode.host 文章汇总:https://binghe.gitcode.host/md/all ...
- Codeforces Round 882 div.2 B
Smiling & Weeping ----玫瑰花你拿才好看,风景要和你看才浪漫--<-<-<@ B. Hamon Odyssey time limit per test 1 ...
- web组态软件(BY组态)介绍
BY组态是什么? BY组态面向工业物联网系统复杂的功能要求,通过"搭积木"的方式,拖拽组件到画布上,实现工业物联网可视化的web开发系统. BY组态适用领域 能源电力.物联网.智能 ...
- 介绍五个很实用的IDEA使用技巧
日常开发中,相信广大 Java 开发者都使用过 IntelliJ IDEA 作为开发工具,IntelliJ IDEA 是一款优秀的 Java 集成开发环境,它提供了许多强大的功能和快捷键,可以帮助开发 ...
- 使用Vue3+elementPlus的Tree组件实现一个拖拽文件夹管理
目录 1.前言 2.分析 3. 实现 4.踩坑 4.1.拖拽辅助线的坑 4.2.数据的坑 4.3.限制拖拽 4.4.样式调整 1.前言 最近在做一个文件夹管理的功能,要实现一个树状的文件夹面板.里面包 ...