当前位置: 首页 > news >正文

有关于做茗茶的网站简单干净的网站

有关于做茗茶的网站,简单干净的网站,手工制作大全图片教程,wordpress首页关键字目录 1. ES分词器详解 1.1 基本概念 1.2 分词发生时期 1.3 分词器的组成 切词器#xff1a;Tokenizer 词项过滤器#xff1a;Token Filter 字符过滤器#xff1a;Character Filter 1.4 倒排索引的数据结构 2. 相关性详解 2.1 什么是相关性#xff08;RelevanceTokenizer 词项过滤器Token Filter 字符过滤器Character Filter 1.4 倒排索引的数据结构 2. 相关性详解 2.1 什么是相关性Relevance 2.2 相关性算法 TF-IDF BM25 2.3 通过Explain API查看TF-IDF 2.4 Boosting Query(常用) 利用must not排除不是苹果公司产品的文档 利用negative_boost降低相关性 3. 单字符串多字段查询 3.1 最佳字段查询Dis Max Query 使用最佳字段查询dis max query 可以通过tie_breaker参数调整 3.2 Multi Match Query(常用) 最佳字段(Best Fields)搜索 使用多数字段Most Fields搜索 3.3 跨字段Cross Field搜索 4. ElasticSearch聚合操作 使用场景 基本语法 4.1 聚合的分类 4.2 Metric Aggregation 4.3 Bucket Aggregation 获取job的分类信息 限定聚合范围 Range Histogram聚合 4.4 Pipeline Aggregation min_bucket示例 Stats示例 percentiles示例 Cumulative_sum示例 4.5 聚合的作用范围 4.6 排序 4.7 ES聚合分析不精准原因分析 4.8 Elasticsearch 聚合性能优化 启用 eager global ordinals 提升高基数聚合性能 插入数据时对索引进行预排序 使用节点查询缓存 使用分片请求缓存 拆分聚合使聚合并行化 1. ES分词器详解 1.1 基本概念 分词器官方称之为文本分析器顾名思义是对文本进行分析处理的一种手段基本处理逻辑为按照预先制定的分词规则把原始文档分割成若干更小粒度的词项粒度大小取决于分词器规则。 1.2 分词发生时期 分词器的处理过程发生在 Index Time 和 Search Time 两个时期。 Index Time文档写入并创建倒排索引时期其分词逻辑取决于映射参数analyzer。Search Time搜索发生时期其分词仅对搜索词产生作用。 1.3 分词器的组成 切词器Tokenizer用于定义切词分词逻辑词项过滤器Token Filter用于对分词之后的单个词项的处理逻辑字符过滤器Character Filter用于处理单个字符 注意 分词器不会对源数据造成任何影响分词仅仅是对倒排索引或者搜索词的行为。 切词器Tokenizer tokenizer 是分词器的核心组成部分之一其主要作用是分词或称之为切词。主要用来对原始文本进行细粒度拆分。拆分之后的每一个部分称之为一个 Term或称之为一个词项。可以把切词器理解为预定义的切词规则。官方内置了很多种切词器默认的切词器为 standard。 词项过滤器Token Filter 词项过滤器用来处理切词完成之后的词项例如把大小写转换删除停用词或同义词处理等。官方同样预置了很多词项过滤器基本可以满足日常开发的需要。当然也是支持第三方也自行开发的。 GET _analyze {filter : [lowercase],text : WWW ELASTIC ORG CN }GET _analyze {tokenizer : standard,filter : [uppercase],text : [www.elastic.org.cn,www elastic org cn] } 运行结果 {tokens : [{token : www elastic org cn,start_offset : 0,end_offset : 18,type : word,position : 0}] }{tokens : [{token : WWW.ELASTIC.ORG.CN,start_offset : 0,end_offset : 18,type : ALPHANUM,position : 0},{token : WWW,start_offset : 19,end_offset : 22,type : ALPHANUM,position : 101},{token : ELASTIC,start_offset : 23,end_offset : 30,type : ALPHANUM,position : 102},{token : ORG,start_offset : 31,end_offset : 34,type : ALPHANUM,position : 103},{token : CN,start_offset : 35,end_offset : 37,type : ALPHANUM,position : 104}] } 停用词 在切词完成之后会被干掉词项即停用词。停用词可以自定义 英文停用词englisha, an, and, are, as, at, be, but, by, for, if, in, into, is, it, no, not, of, on, or, such, that, the, their, then, there, these, they, this, to, was, will, with。 中日韩停用词cjka, and, are, as, at, be, but, by, for, if, in, into, is, it, no, not, of, on, or, s, such, t, that, the, their, then, there, these, they, this, to, was, will, with, www。 GET _analyze {tokenizer: standard, filter: [stop],text: [What are you doing] }### 自定义 filter DELETE test_token_filter_stop PUT test_token_filter_stop {settings: {analysis: {filter: {my_filter: {type: stop,stopwords: [www],ignore_case: true}}}} } GET test_token_filter_stop/_analyze {tokenizer: standard, filter: [my_filter], text: [What www WWW are you doing] } 运行结果 第一个GET {tokens : [{token : What,start_offset : 0,end_offset : 4,type : ALPHANUM,position : 0},{token : you,start_offset : 9,end_offset : 12,type : ALPHANUM,position : 2},{token : doing,start_offset : 13,end_offset : 18,type : ALPHANUM,position : 3}] }第二个GET {tokens : [{token : What,start_offset : 0,end_offset : 4,type : ALPHANUM,position : 0},{token : are,start_offset : 13,end_offset : 16,type : ALPHANUM,position : 3},{token : you,start_offset : 17,end_offset : 20,type : ALPHANUM,position : 4},{token : doing,start_offset : 21,end_offset : 26,type : ALPHANUM,position : 5}] } 同义词 同义词定义规则 a, b, c d这种方式a、b、c 会被 d 代替。a, b, c, d这种方式下a、b、c、d 是等价的。 PUT test_token_filter_synonym {settings: {analysis: {filter: {my_synonym: {type: synonym,synonyms: [ good, nice excellent ] //good, nice, excellent}}}} } GET test_token_filter_synonym/_analyze {tokenizer: standard, filter: [my_synonym], text: [good] } 运行结果 {tokens : [{token : excellent,start_offset : 0,end_offset : 4,type : SYNONYM,position : 0}] } 字符过滤器Character Filter 分词之前的预处理过滤无用字符。 PUT index_name {settings: {analysis: {char_filter: {my_char_filter: {type: char_filter_type}}}} } type使用的字符过滤器类型名称可配置以下值 html_stripmappingpattern_replace HTML 标签过滤器HTML Strip Character Filter 字符过滤器会去除 HTML 标签和转义 HTML 元素如 、 PUT test_html_strip_filter {settings: {analysis: {char_filter: {my_char_filter: {type: html_strip, // html_strip 代表使用 HTML 标签过滤器escaped_tags: [ // 当前仅保留 a 标签 a]}}}} } GET test_html_strip_filter/_analyze {tokenizer: standard, char_filter: [my_char_filter],text: [pIapos;m so ahappy/a!/p] } 运行结果 {tokens : [{token : Im,start_offset : 3,end_offset : 11,type : ALPHANUM,position : 0},{token : so,start_offset : 12,end_offset : 14,type : ALPHANUM,position : 1},{token : a,start_offset : 16,end_offset : 17,type : ALPHANUM,position : 2},{token : happy,start_offset : 18,end_offset : 23,type : ALPHANUM,position : 3},{token : a,start_offset : 25,end_offset : 26,type : ALPHANUM,position : 4}] } 参数escaped_tags需要保留的 html 标签 字符映射过滤器Mapping Character Filter 通过定义映射替换为规则把特定字符替换为指定字符 PUT test_html_strip_filter {settings: {analysis: {char_filter: {my_char_filter: {type: mapping, // mapping 代表使用字符映射过滤器mappings: [ // 数组中规定的字符会被等价替换为 指定的字符滚 *,垃 *,圾 *]}}}} } GET test_html_strip_filter/_analyze {//tokenizer: standard, char_filter: [my_char_filter],text: 你就是个垃圾滚 } 运行结果 {tokens : [{token : 你就是个***,start_offset : 0,end_offset : 8,type : word,position : 0}] } 正则替换过滤器Pattern Replace Character Filter PUT text_pattern_replace_filter {settings: {analysis: {char_filter: {my_char_filter: {type: pattern_replace, // pattern_replace 代表使用正则替换过滤器 pattern: (\d{3})\d{4}(\d{4}), // 正则表达式replacement: $1****$2}}}} } GET text_pattern_replace_filter/_analyze {char_filter: [my_char_filter],text: 您的手机号是18868686688 } 运行结果 {tokens : [{token : 您的手机号是188****6688,start_offset : 0,end_offset : 17,type : word,position : 0}] } 1.4 倒排索引的数据结构 当数据写入 ES 时数据将会通过 分词 被切分为不同的 termES 将 term 与其对应的文档列表建立一种映射关系这种结构就是 倒排索引。 如下图所示 为了进一步提升索引的效率ES 在 term 的基础上利用 term 的前缀或者后缀构建了 term index, 用于对 term 本身进行索引ES 实际的索引结构如下图所示 这样当我们去搜索某个关键词时ES 首先根据它的前缀或者后缀迅速缩小关键词的在 term dictionary 中的范围大大减少了磁盘IO的次数。 单词词典Term Dictionary) 记录所有文档的单词记录单词到倒排列表的关联关系 常用字典数据结构lucene字典实现原理 - zhanlijun - 博客园 倒排列表(Posting List)-记录了单词对应的文档结合由倒排索引项组成倒排索引项(Posting) 文档ID词频TF–该单词在文档中出现的次数用于相关性评分位置(Position)-单词在文档中分词的位置。用于短语搜索match phrase query)偏移(Offset)-记录单词的开始结束位置实现高亮显示 Elasticsearch 的JSON文档中的每个字段都有自己的倒排索引。 可以指定对某些字段不做索引 优点︰节省存储空间缺点: 字段无法被搜索 2. 相关性详解 搜索是用户和搜索引擎的对话用户关心的是搜索结果的相关性 是否可以找到所有相关的内容有多少不相关的内容被返回了文档的打分是否合理结合业务需求平衡结果排名 2.1 什么是相关性Relevance 搜索的相关性算分描述了一个文档和查询语句匹配的程度。ES 会对每个匹配查询条件的结果进行算分_score。打分的本质是排序需要把最符合用户需求的文档排在前面。 如下例子显而易见查询JAVA多线程设计模式文档id为23的文档的算分更高 关键词 文档ID JAVA 123 设计模式 123456 多线程 2379 如何衡量相关性 Precision(查准率)―尽可能返回较少的无关文档Recall(查全率)–尽量返回较多的相关文档Ranking -是否能够按照相关度进行排序 2.2 相关性算法 ES 5之前默认的相关性算分采用TF-IDF现在采用BM 25。 TF-IDF TF-IDFterm frequency–inverse document frequency是一种用于信息检索与数据挖掘的常用加权技术。 TF-IDF被公认为是信息检索领域最重要的发明除了在信息检索在文献分类和其他相关领域有着非常广泛的应用。IDF的概念最早是剑桥大学的“斯巴克.琼斯”提出 1972年——“关键词特殊性的统计解释和它在文献检索中的应用”但是没有从理论上解释IDF应该是用log(全部文档数/检索词出现过的文档总数)而不是其他函数也没有做进一步的研究19701980年代萨尔顿和罗宾逊进行了进一步的证明和研究并用香农信息论做了证明http://www.staff.city.ac.uk/~sb317/papers/foundations_bm25_review.pdf 现代搜索引擎对TF-IDF进行了大量细微的优化 Lucene中的TF-IDF评分公式 TF是词频(Term Frequency) 检索词在文档中出现的频率越高相关性也越高。 词频TF 某个词在文档中出现的次数 / 文档的总词数 IDF是逆向文本频率(Inverse Document Frequency) 每个检索词在索引中出现的频率频率越高相关性越低。总文档中有些词比如“是”、“的” 、“在” 在所有文档中出现频率都很高并不重要可以减少多个文档中都频繁出现的词的权重。 逆向文本频率IDF log (语料库的文档总数 / (包含该词的文档数1)) 字段长度归一值 field-length norm 检索词出现在一个内容短的 title 要比同样的词出现在一个内容长的 content 字段权重更大。 以上三个因素——词频term frequency、逆向文本频率inverse document frequency和字段长度归一值field-length norm——是在索引时计算并存储的最后将它们结合在一起计算单个词在特定文档中的权重。 BM25 BM25 就是对 TF-IDF 算法的改进对于 TF-IDF 算法TF(t) 部分的值越大整个公式返回的值就会越大。BM25 就针对这点进行来优化随着TF(t) 的逐步加大该算法的返回值会趋于一个数值。 从ES 5开始默认算法改为BM 25和经典的TF-IDF相比当TF无限增加时BM 25算分会趋于一个数值 BM 25的公式 2.3 通过Explain API查看TF-IDF PUT /test_score/_bulk {index:{_id:1}} {content:we use Elasticsearch to power the search} {index:{_id:2}} {content:we like elasticsearch} {index:{_id:3}} {content:Thre scoring of documents is caculated by the scoring formula} {index:{_id:4}} {content:you know,for search}GET /test_score/_search {explain: true, query: {match: {content: elasticsearch}} }GET /test_score/_explain/2 {query: {match: {content: elasticsearch}} } 运行结果 {_index : test_score,_type : _doc,_id : 2,matched : true,explanation : {value : 0.8713851,description : weight(content:elasticsearch in 1) [PerFieldSimilarity], result of:,details : [{value : 0.8713851,description : score(freq1.0), computed as boost * idf * tf from:,details : [{value : 2.2,description : boost,details : [ ]},{value : 0.6931472,description : idf, computed as log(1 (N - n 0.5) / (n 0.5)) from:,details : [{value : 2,description : n, number of documents containing term,details : [ ]},{value : 4,description : N, total number of documents with field,details : [ ]}]},{value : 0.5714286,description : tf, computed as freq / (freq k1 * (1 - b b * dl / avgdl)) from:,details : [{value : 1.0,description : freq, occurrences of term within document,details : [ ]},{value : 1.2,description : k1, term saturation parameter,details : [ ]},{value : 0.75,description : b, length normalization parameter,details : [ ]},{value : 3.0,description : dl, length of field,details : [ ]},{value : 6.0,description : avgdl, average length of field,details : [ ]}]}]}]} } GET /es_db/_explain/3 {query: {match: {address: 广州公园}} } 运行结果 {_index : es_db,_type : _doc,_id : 3,matched : true,explanation : {value : 1.6476591,description : sum of:,details : [{value : 0.5978369,description : weight(address:广州 in 2) [PerFieldSimilarity], result of:,details : [{value : 0.5978369,description : score(freq1.0), computed as boost * idf * tf from:,details : [{value : 2.2,description : boost,details : [ ]},{value : 0.597837,description : idf, computed as log(1 (N - n 0.5) / (n 0.5)) from:,details : [{value : 5,description : n, number of documents containing term,details : [ ]},{value : 9,description : N, total number of documents with field,details : [ ]}]},{value : 0.45454544,description : tf, computed as freq / (freq k1 * (1 - b b * dl / avgdl)) from:,details : [{value : 1.0,description : freq, occurrences of term within document,details : [ ]},{value : 1.2,description : k1, term saturation parameter,details : [ ]},{value : 0.75,description : b, length normalization parameter,details : [ ]},{value : 5.0,description : dl, length of field,details : [ ]},{value : 5.0,description : avgdl, average length of field,details : [ ]}]}]}]},{value : 1.0498221,description : weight(address:公园 in 2) [PerFieldSimilarity], result of:,details : [{value : 1.0498221,description : score(freq1.0), computed as boost * idf * tf from:,details : [{value : 2.2,description : boost,details : [ ]},{value : 1.0498221,description : idf, computed as log(1 (N - n 0.5) / (n 0.5)) from:,details : [{value : 3,description : n, number of documents containing term,details : [ ]},{value : 9,description : N, total number of documents with field,details : [ ]}]},{value : 0.45454544,description : tf, computed as freq / (freq k1 * (1 - b b * dl / avgdl)) from:,details : [{value : 1.0,description : freq, occurrences of term within document,details : [ ]},{value : 1.2,description : k1, term saturation parameter,details : [ ]},{value : 0.75,description : b, length normalization parameter,details : [ ]},{value : 5.0,description : dl, length of field,details : [ ]},{value : 5.0,description : avgdl, average length of field,details : [ ]}]}]}]}]} } 2.4 Boosting Query(常用) Boosting是控制相关度的一种手段。可以通过指定字段的boost值影响查询结果 参数boost的含义 当boost 1时打分的权重相对性提升当0 boost 1时打分的权重相对性降低当boost 0时贡献负分 应用场景希望包含了某项内容的结果不是不出现而是排序靠后。 POST /blogs/_bulk {index:{_id:1}} {title:Apple iPad,content:Apple iPad,Apple iPad} {index:{_id:2}} {title:Apple iPad,Apple iPad,content:Apple iPad}GET /blogs/_search {query: {bool: {should: [{match: {title: {query: apple,ipad,boost: 1}}},{match: {content: {query: apple,ipad,boost: 4}}}]}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 2.2558527,hits : [{_index : blogs,_type : _doc,_id : 1,_score : 2.2558527,_source : {title : Apple iPad,content : Apple iPad,Apple iPad}},{_index : blogs,_type : _doc,_id : 2,_score : 2.1472821,_source : {title : Apple iPad,Apple iPad,content : Apple iPad}}]} } 案例要求苹果公司的产品信息优先展示 POST /news/_bulk {index:{_id:1}} {content:Apple Mac} {index:{_id:2}} {content:Apple iPad} {index:{_id:3}} {content:Apple employee like Apple Pie and Apple Juice}GET /news/_search {query: {bool: {must: {match: {content: apple}}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 3,relation : eq},max_score : 0.17280531,hits : [{_index : news,_type : _doc,_id : 3,_score : 0.17280531,_source : {content : Apple employee like Apple Pie and Apple Juice}},{_index : news,_type : _doc,_id : 1,_score : 0.16786805,_source : {content : Apple Mac}},{_index : news,_type : _doc,_id : 2,_score : 0.16786805,_source : {content : Apple iPad}}]} } 利用must not排除不是苹果公司产品的文档 GET /news/_search {query: {bool: {must: {match: {content: apple}},must_not: {match:{content: pie}}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 0.16786805,hits : [{_index : news,_type : _doc,_id : 1,_score : 0.16786805,_source : {content : Apple Mac}},{_index : news,_type : _doc,_id : 2,_score : 0.16786805,_source : {content : Apple iPad}}]} } 利用negative_boost降低相关性 对某些返回结果不满意但又不想排除掉 must_not)可以考虑boosting query的negative_boost。 negative_boost 对 negative部分query生效计算评分时boosting部分评分不修改negative部分query乘以negative_boost值negative_boost取值:0-1.0举例:0.3 GET /news/_search {query: {boosting: {positive: {match: {content: apple}},negative: {match: {content: pie}},negative_boost: 0.2}} } 运行结果 {took : 11,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 3,relation : eq},max_score : 0.16786805,hits : [{_index : news,_type : _doc,_id : 1,_score : 0.16786805,_source : {content : Apple Mac}},{_index : news,_type : _doc,_id : 2,_score : 0.16786805,_source : {content : Apple iPad}},{_index : news,_type : _doc,_id : 3,_score : 0.034561064,_source : {content : Apple employee like Apple Pie and Apple Juice}}]} } 3. 单字符串多字段查询 三种场景 最佳字段(Best Fields) 当字段之间相互竞争又相互关联。例如对于博客的 title和 body这样的字段评分来自最匹配字段 多数字段(Most Fields) 处理英文内容时的一种常见的手段是在主字段( English Analyzer)抽取词干加入同义词以匹配更多的文档。相同的文本加入子字段Standard Analyzer以提供更加精确的匹配。其他字段作为匹配文档提高相关度的信号匹配字段越多则越好。 混合字段(Cross Fields) 对于某些实体例如人名地址图书信息。需要在多个字段中确定信息单个字段只能作为整体的一部分。希望在任何这些列出的字段中找到尽可能多的词。 3.1 最佳字段查询Dis Max Query 将任何与任一查询匹配的文档作为结果返回采用字段上最匹配的评分最终评分返回。 max(a,b) 官方文档Disjunction max query | Elasticsearch Guide [7.17] | Elastic 测试 DELETE /blogs PUT /blogs/_doc/1 {title: Quick brown rabbits,body: Brown rabbits are commonly seen. }PUT /blogs/_doc/2 {title: Keeping pets healthy,body: My quick brown fox eats rabbits on a regular basis. }POST /blogs/_search {query: {bool: {should: [{ match: { title: Brown fox }},{ match: { body: Brown fox }}]}} } 运行结果 {took : 701,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 0.90425634,hits : [{_index : blogs,_type : _doc,_id : 1,_score : 0.90425634,_source : {title : Quick brown rabbits,body : Brown rabbits are commonly seen.}},{_index : blogs,_type : _doc,_id : 2,_score : 0.77041256,_source : {title : Keeping pets healthy,body : My quick brown fox eats rabbits on a regular basis.}}]} } 思考查询结果不符合预期为什么 bool should的算法过程 查询should语句中的两个查询加和两个查询的评分乘以匹配语句的总数除以所有语句的总数 上述例子中title和body属于竞争关系不应该将分数简单叠加而是应该找到单个最佳匹配的字段的评分。 使用最佳字段查询dis max query POST /blogs/_search {query: {dis_max: {queries: [{ match: { title: Brown fox }},{ match: { body: Brown fox }}]}} } 运行结果-符合预期 {took : 6,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 0.77041256,hits : [{_index : blogs,_type : _doc,_id : 2,_score : 0.77041256,_source : {title : Keeping pets healthy,body : My quick brown fox eats rabbits on a regular basis.}},{_index : blogs,_type : _doc,_id : 1,_score : 0.6931471,_source : {title : Quick brown rabbits,body : Brown rabbits are commonly seen.}}]} } 可以通过tie_breaker参数调整 Tier Breaker是一个介于0-1之间的浮点数。0代表使用最佳匹配1代表所有语句同等重要。 获得最佳匹配语句的评分_score 。将其他匹配语句的评分与tie_breaker相乘对以上评分求和并规范化 最终得分最佳匹配字段其他匹配字段*tie_breaker POST /blogs/_search {query: {dis_max: {queries: [{ match: { title: Brown fox }},{ match: { body: Brown fox }}],tie_breaker: 0.1}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 0.77041256,hits : [{_index : blogs,_type : _doc,_id : 2,_score : 0.77041256,_source : {title : Keeping pets healthy,body : My quick brown fox eats rabbits on a regular basis.}},{_index : blogs,_type : _doc,_id : 1,_score : 0.714258,_source : {title : Quick brown rabbits,body : Brown rabbits are commonly seen.}}]} } 3.2 Multi Match Query(常用) 最佳字段(Best Fields)搜索 best_fields策略获取最佳匹配字段的得分, final_score max(其他匹配字段得分 最佳匹配字段得分) 采用 best_fields 查询并添加参数 tie_breaker0.1final_score 其他匹配字段得分 * 0.1 最佳匹配字段得分 Best Fields是默认类型可以不用指定等价于dis_max查询方式 POST /blogs/_search {query: {multi_match: {type: best_fields,query: Brown fox,fields: [title,body],tie_breaker: 0.2}} } 运行结果 {took : 2,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 0.77041256,hits : [{_index : blogs,_type : _doc,_id : 2,_score : 0.77041256,_source : {title : Keeping pets healthy,body : My quick brown fox eats rabbits on a regular basis.}},{_index : blogs,_type : _doc,_id : 1,_score : 0.73536897,_source : {title : Quick brown rabbits,body : Brown rabbits are commonly seen.}}]} } 案例 PUT /employee {settings : {index : {analysis.analyzer.default.type: ik_max_word}} }POST /employee/_bulk {index:{_id:1}} {empId:1,name:员工001,age:20,sex:男,mobile:19000001111,salary:23343,deptName:技术部,address:湖北省武汉市洪山区光谷大厦,content:i like to write best elasticsearch article} {index:{_id:2}} {empId:2,name:员工002,age:25,sex:男,mobile:19000002222,salary:15963,deptName:销售部,address:湖北省武汉市江汉路,content:i think java is the best programming language} {index:{_id:3}} {empId:3,name:员工003,age:30,sex:男,mobile:19000003333,salary:20000,deptName:技术部,address:湖北省武汉市经济开发区,content:i am only an elasticsearch beginner} {index:{_id:4}} {empId:4,name:员工004,age:20,sex:女,mobile:19000004444,salary:15600,deptName:销售部,address:湖北省武汉市沌口开发区,content:elasticsearch and hadoop are all very good solution, i am a beginner} {index:{_id:5}} {empId:5,name:员工005,age:20,sex:男,mobile:19000005555,salary:19665,deptName:测试部,address:湖北省武汉市东湖隧道,content:spark is best big data solution based on scala, an programming language similar to java} {index:{_id:6}} {empId:6,name:员工006,age:30,sex:女,mobile:19000006666,salary:30000,deptName:技术部,address:湖北省武汉市江汉路,content:i like java developer} {index:{_id:7}} {empId:7,name:员工007,age:60,sex:女,mobile:19000007777,salary:52130,deptName:测试部,address:湖北省黄冈市边城区,content:i like elasticsearch developer} {index:{_id:8}} {empId:8,name:员工008,age:19,sex:女,mobile:19000008888,salary:60000,deptName:技术部,address:湖北省武汉市江汉大学,content:i like spark language} {index:{_id:9}} {empId:9,name:员工009,age:40,sex:男,mobile:19000009999,salary:23000,deptName:销售部,address:河南省郑州市郑州大学,content:i like java developer} {index:{_id:10}} {empId:10,name:张湖北,age:35,sex:男,mobile:19000001010,salary:18000,deptName:测试部,address:湖北省武汉市东湖高新,content:i like java developer, i also like elasticsearch} {index:{_id:11}} {empId:11,name:王河南,age:61,sex:男,mobile:19000001011,salary:10000,deptName:销售部,address:河南省开封市河南大学,content:i am not like java} {index:{_id:12}} {empId:12,name:张大学,age:26,sex:女,mobile:19000001012,salary:11321,deptName:测试部,address:河南省开封市河南大学,content:i am java developer, java is good} {index:{_id:13}} {empId:13,name:李江汉,age:36,sex:男,mobile:19000001013,salary:11215,deptName:销售部,address:河南省郑州市二七区,content:i like java and java is very best, i like it, do you like java} {index:{_id:14}} {empId:14,name:王技术,age:45,sex:女,mobile:19000001014,salary:16222,deptName:测试部,address:河南省郑州市金水区,content:i like c} {index:{_id:15}} {empId:15,name:张测试,age:18,sex:男,mobile:19000001015,salary:20000,deptName:技术部,address:河南省郑州市高新开发区,content:i think spark is good}GET /employee/_search {query: {multi_match: {query: elasticsearch beginner 湖北省 开封市,type: best_fields,fields: [content,address]}},size: 15 }# 查看执行计划 GET /employee/_explain/3 {query: {multi_match: {query: elasticsearch beginner 湖北省 开封市,type: best_fields,fields: [content,address]}} }GET /employee/_explain/3 {query: {multi_match: {query: elasticsearch beginner 湖北省 开封市,type: best_fields,fields: [content,address],tie_breaker: 0.1}} } 使用多数字段Most Fields搜索 most_fields策略获取全部匹配字段的累计得分综合全部匹配字段的得分等价于bool should查询方式 GET /employee/_explain/3 {query: {multi_match: {query: elasticsearch beginner 湖北省 开封市,type: most_fields,fields: [content,address]}} } 案例 DELETE /titles PUT /titles {mappings: {properties: {title: {type: text,analyzer: english,fields: {std: {type: text,analyzer: standard}}}}} }POST titles/_bulk { index: { _id: 1 }} { title: My dog barks } { index: { _id: 2 }} { title: I see a lot of barking dogs on the road }# 结果与预期不匹配 GET /titles/_search {query: {match: {title: barking dogs}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 0.42221838,hits : [{_index : titles,_type : _doc,_id : 1,_score : 0.42221838,_source : {title : My dog barks}},{_index : titles,_type : _doc,_id : 2,_score : 0.320886,_source : {title : I see a lot of barking dogs on the road }}]} } 用广度匹配字段title包括尽可能多的文档——以提升召回率——同时又使用字段title.std 作为信号将相关度更高的文档置于结果顶部。 GET /titles/_search {query: {multi_match: {query: barking dogs,type: most_fields,fields: [title,title.std]}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 1.4569323,hits : [{_index : titles,_type : _doc,_id : 2,_score : 1.4569323,_source : {title : I see a lot of barking dogs on the road }},{_index : titles,_type : _doc,_id : 1,_score : 0.42221838,_source : {title : My dog barks}}]} } 每个字段对于最终评分的贡献可以通过自定义值boost 来控制。比如使 title 字段更为重要这样同时也降低了其他信号字段的作用 #增加title的权重 GET /titles/_search {query: {multi_match: {query: barking dogs,type: most_fields,fields: [title^10,title.std]}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 2,relation : eq},max_score : 4.3449063,hits : [{_index : titles,_type : _doc,_id : 2,_score : 4.3449063,_source : {title : I see a lot of barking dogs on the road }},{_index : titles,_type : _doc,_id : 1,_score : 4.222184,_source : {title : My dog barks}}]} } 3.3 跨字段Cross Field搜索 搜索内容在多个字段中都显示类似booldis_max组合 DELETE /address PUT /address {settings : {index : {analysis.analyzer.default.type: ik_max_word}} }PUT /address/_bulk { index: { _id: 1} } {province: 湖南,city: 长沙} { index: { _id: 2} } {province: 湖南,city: 常德} { index: { _id: 3} } {province: 广东,city: 广州} { index: { _id: 4} } {province: 湖南,city: 邵阳}#使用most_fields的方式结果不符合预期不支持operator GET /address/_search {query: {multi_match: {query: 湖南常德,type: most_fields,fields: [province,city]}} }# 可以使用cross_fields支持operator #与copy_to相比其中一个优势就是它可以在搜索时为单个字段提升权重。 GET /address/_search {query: {multi_match: {query: 湖南常德,type: cross_fields,operator: and, fields: [province,city]}} } 运行结果 第一个GET {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 3,relation : eq},max_score : 1.5606477,hits : [{_index : address,_type : _doc,_id : 2,_score : 1.5606477,_source : {province : 湖南,city : 常德}},{_index : address,_type : _doc,_id : 1,_score : 0.35667494,_source : {province : 湖南,city : 长沙}},{_index : address,_type : _doc,_id : 4,_score : 0.35667494,_source : {province : 湖南,city : 邵阳}}]} }第二个GET {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 1,relation : eq},max_score : 1.5606477,hits : [{_index : address,_type : _doc,_id : 2,_score : 1.5606477,_source : {province : 湖南,city : 常德}}]} } 可以用copy...to 解决但是需要额外的存储空间 DELETE /address # copy_to参数允许将多个字段的值复制到组字段中然后可以将其作为单个字段进行查询 PUT /address {mappings : {properties : {province : {type : keyword,copy_to: full_address},city : {type : text,copy_to: full_address}}},settings : {index : {analysis.analyzer.default.type: ik_max_word}} }PUT /address/_bulk { index: { _id: 1} } {province: 湖南,city: 长沙} { index: { _id: 2} } {province: 湖南,city: 常德} { index: { _id: 3} } {province: 广东,city: 广州} { index: { _id: 4} } {province: 湖南,city: 邵阳}GET /address/_search {query: {match: {full_address: {query: 湖南常德,operator: and}}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 1,relation : eq},max_score : 1.5606477,hits : [{_index : address,_type : _doc,_id : 2,_score : 1.5606477,_source : {province : 湖南,city : 常德}}]} } 4. ElasticSearch聚合操作 Elasticsearch除搜索以外提供了针对ES 数据进行统计分析的功能。聚合(aggregations)可以让我们极其方便的实现对数据的统计、分析、运算。例如 什么品牌的手机最受欢迎这些手机的平均价格、最高价格、最低价格这些手机每月的销售情况如何 使用场景 聚合查询可以用于各种场景比如商业智能、数据挖掘、日志分析等等。 电商平台的销售分析统计每个地区的销售额、每个用户的消费总额、每个产品的销售量等以便更好地了解销售情况和趋势。社交媒体的用户行为分析统计每个用户的发布次数、转发次数、评论次数等以便更好地了解用户行为和趋势同时可以将数据按照地区、时间、话题等维度进行分析。物流企业的运输分析统计每个区域的运输量、每个车辆的运输次数、每个司机的行驶里程等以便更好地了解运输情况和优化运输效率。金融企业的交易分析统计每个客户的交易总额、每个产品的销售量、每个交易员的业绩等以便更好地了解交易情况和优化业务流程。智能家居的设备监控分析统计每个设备的使用次数、每个家庭的能源消耗量、每个时间段的设备使用率等以便更好地了解用户需求和优化设备效能。 基本语法 聚合查询的语法结构与其他查询相似通常包含以下部分 查询条件指定需要聚合的文档可以使用标准的 Elasticsearch 查询语法如 term、match、range 等等。聚合函数指定要执行的聚合操作如 sum、avg、min、max、terms、date_histogram 等等。每个聚合命令都会生成一个聚合结果。聚合嵌套聚合命令可以嵌套以便更细粒度地分析数据。 GET index_name/_search {aggs: {aggs_name: { // 聚合名称需要自己定义agg_type: {field: field_name}}} } aggs_name聚合函数的名称agg_type聚合种类比如是桶聚合terms或者是指标聚合avg、sum、min、max等field_name字段名称或者叫域名。 4.1 聚合的分类 Metric Aggregation—些数学运算可以对文档字段进行统计分析类比Mysql中的 min(), max(), sum() 操作。 SELECT MIN(price), MAX(price) FROM products #Metric聚合的DSL类比实现 {aggs:{avg_price:{avg:{field:price}}} } Bucket Aggregation 一些满足特定条件的文档的集合放置到一个桶里每一个桶关联一个key类比Mysql中的group by操作。 SELECT size COUNT(*) FROM products GROUP BY size #bucket聚合的DSL类比实现 {aggs: {by_size: {terms: {field: size}} } Pipeline Aggregation对其他的聚合结果进行二次聚合 示例数据 DELETE /employees #创建索引库 PUT /employees {mappings: {properties: {age:{type: integer},gender:{type: keyword},job:{type : text,fields : {keyword : {type : keyword,ignore_above : 50}}},name:{type: keyword},salary:{type: integer}}} }PUT /employees/_bulk { index : { _id : 1 } } { name : Emma,age:32,job:Product Manager,gender:female,salary:35000 } { index : { _id : 2 } } { name : Underwood,age:41,job:Dev Manager,gender:male,salary: 50000} { index : { _id : 3 } } { name : Tran,age:25,job:Web Designer,gender:male,salary:18000 } { index : { _id : 4 } } { name : Rivera,age:26,job:Web Designer,gender:female,salary: 22000} { index : { _id : 5 } } { name : Rose,age:25,job:QA,gender:female,salary:18000 } { index : { _id : 6 } } { name : Lucy,age:31,job:QA,gender:female,salary: 25000} { index : { _id : 7 } } { name : Byrd,age:27,job:QA,gender:male,salary:20000 } { index : { _id : 8 } } { name : Foster,age:27,job:Java Programmer,gender:male,salary: 20000} { index : { _id : 9 } } { name : Gregory,age:32,job:Java Programmer,gender:male,salary:22000 } { index : { _id : 10 } } { name : Bryant,age:20,job:Java Programmer,gender:male,salary: 9000} { index : { _id : 11 } } { name : Jenny,age:36,job:Java Programmer,gender:female,salary:38000 } { index : { _id : 12 } } { name : Mcdonald,age:31,job:Java Programmer,gender:male,salary: 32000} { index : { _id : 13 } } { name : Jonthna,age:30,job:Java Programmer,gender:female,salary:30000 } { index : { _id : 14 } } { name : Marshall,age:32,job:Javascript Programmer,gender:male,salary: 25000} { index : { _id : 15 } } { name : King,age:33,job:Java Programmer,gender:male,salary:28000 } { index : { _id : 16 } } { name : Mccarthy,age:21,job:Javascript Programmer,gender:male,salary: 16000} { index : { _id : 17 } } { name : Goodwin,age:25,job:Javascript Programmer,gender:male,salary: 16000} { index : { _id : 18 } } { name : Catherine,age:29,job:Javascript Programmer,gender:female,salary: 20000} { index : { _id : 19 } } { name : Boone,age:30,job:DBA,gender:male,salary: 30000} { index : { _id : 20 } } { name : Kathy,age:29,job:DBA,gender:female,salary: 20000} 4.2 Metric Aggregation 单值分析︰只输出一个分析结果 min, max, avg, sumCardinality类似distinct Count)多值分析:输出多个分析结果stats统计, extended statspercentile 百分位, percentile ranktop hits(排在前面的示例) 查询员工的最低最高和平均工资 #多个 Metric 聚合找到最低最高和平均工资 POST /employees/_search {size: 0, aggs: {max_salary: {max: {field: salary}},min_salary: {min: {field: salary}},avg_salary: {avg: {field: salary}}} } 运行结果 {took : 5,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {max_salary : {value : 50000.0},avg_salary : {value : 24700.0},min_salary : {value : 9000.0}} } 对salary进行统计 # 一个聚合输出多值 POST /employees/_search {size: 0,aggs: {stats_salary: {stats: {field:salary}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {stats_salary : {count : 20,min : 9000.0,max : 50000.0,avg : 24700.0,sum : 494000.0}} } cardinate对搜索结果去重 POST /employees/_search {size: 0,aggs: {cardinate: {cardinality: {field: job.keyword}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {cardinate : {value : 7}} } 4.3 Bucket Aggregation 按照一定的规则将文档分配到不同的桶中从而达到分类的目的。ES提供的一些常见的 Bucket Aggregation。 Terms需要字段支持filedata keyword 默认支持fielddatatext需要在Mapping 中开启fielddata会按照分词后的结果进行分桶 数字类型 Range / Data RangeHistogram直方图 / Date Histogram 支持嵌套: 也就在桶里再做分桶 桶聚合可以用于各种场景例如 对数据进行分组统计比如按照地区、年龄段、性别等字段进行分组统计。对时间序列数据进行时间段分析比如按照每小时、每天、每月、每季度、每年等时间段进行分析。对各种标签信息分类并统计其数量。 获取job的分类信息 # 对keword 进行聚合 GET /employees/_search {size: 0,aggs: {jobs: {terms: {field:job.keyword}}} } 运行结果 {took : 6,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7},{key : Javascript Programmer,doc_count : 4},{key : QA,doc_count : 3},{key : DBA,doc_count : 2},{key : Web Designer,doc_count : 2},{key : Dev Manager,doc_count : 1},{key : Product Manager,doc_count : 1}]}} } 聚合可配置属性有 field指定聚合字段size指定聚合结果数量order指定聚合结果排序方式 默认情况下Bucket聚合会统计Bucket内的文档数量记为_count并且按照_count降序排序。 我们可以指定order属性自定义聚合的排序方式 GET /employees/_search {size: 0,aggs: {jobs: {terms: {field:job.keyword,size: 10,order: {_count: desc }}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7},{key : Javascript Programmer,doc_count : 4},{key : QA,doc_count : 3},{key : DBA,doc_count : 2},{key : Web Designer,doc_count : 2},{key : Dev Manager,doc_count : 1},{key : Product Manager,doc_count : 1}]}} } 限定聚合范围 #只对salary在10000元以上的文档聚合 GET /employees/_search {query: {range: {salary: {gte: 10000 }}}, size: 0,aggs: {jobs: {terms: {field:job.keyword,size: 10,order: {_count: desc }}}} } 运行结果 {took : 2,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 19,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 6},{key : Javascript Programmer,doc_count : 4},{key : QA,doc_count : 3},{key : DBA,doc_count : 2},{key : Web Designer,doc_count : 2},{key : Dev Manager,doc_count : 1},{key : Product Manager,doc_count : 1}]}} } 注意对 Text 字段进行 terms 聚合查询会失败抛出异常 POST /employees/_search {size: 0,aggs: {jobs: {terms: {field:job}}} } 解决办法对 Text 字段打开 fielddata支持terms aggregation不推荐 PUT /employees/_mapping {properties : {job:{type: text,fielddata: true}} }# 对 Text 字段进行分词分词后的terms POST /employees/_search {size: 0,aggs: {jobs: {terms: {field:job}}} } 运行结果 {took : 7,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : programmer,doc_count : 11},{key : java,doc_count : 7},{key : javascript,doc_count : 4},{key : qa,doc_count : 3},{key : dba,doc_count : 2},{key : designer,doc_count : 2},{key : manager,doc_count : 2},{key : web,doc_count : 2},{key : dev,doc_count : 1},{key : product,doc_count : 1}]}} } 对job.keyword 和 job 进行 terms 聚合分桶的总数并不一样 POST /employees/_search {size: 0,aggs: {cardinate: {cardinality: {field: job}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {cardinate : {value : 10}} } Range Histogram聚合 按照数字的范围进行分桶在Range Aggregation中可以自定义Key Range 示例按照工资的 Range 分桶 Salary Range分桶可以自己定义 key POST employees/_search {size: 0,aggs: {salary_range: {range: {field:salary,ranges:[{to:10000},{from:10000,to:20000},{key:20000,from:20000}]}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {salary_range : {buckets : [{key : *-10000.0,to : 10000.0,doc_count : 1},{key : 10000.0-20000.0,from : 10000.0,to : 20000.0,doc_count : 4},{key : 20000,from : 20000.0,doc_count : 15}]}} }Histogram示例按照工资的间隔分桶 #工资0到10万以 5000一个区间进行分桶 POST employees/_search {size: 0,aggs: {salary_histrogram: {histogram: {field:salary,interval:5000,extended_bounds:{min:0,max:100000}}}} } 运行结果 {took : 9,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {salary_histrogram : {buckets : [{key : 0.0,doc_count : 0},{key : 5000.0,doc_count : 1},{key : 10000.0,doc_count : 0},{key : 15000.0,doc_count : 4},{key : 20000.0,doc_count : 6},{key : 25000.0,doc_count : 3},{key : 30000.0,doc_count : 3},{key : 35000.0,doc_count : 2},{key : 40000.0,doc_count : 0},{key : 45000.0,doc_count : 0},{key : 50000.0,doc_count : 1},{key : 55000.0,doc_count : 0},{key : 60000.0,doc_count : 0},{key : 65000.0,doc_count : 0},{key : 70000.0,doc_count : 0},{key : 75000.0,doc_count : 0},{key : 80000.0,doc_count : 0},{key : 85000.0,doc_count : 0},{key : 90000.0,doc_count : 0},{key : 95000.0,doc_count : 0},{key : 100000.0,doc_count : 0}]}} } top_hits应用场景: 当获取分桶后桶内最匹配的顶部文档列表 # 指定size不同工种中年纪最大的3个员工的具体信息 POST /employees/_search {size: 0,aggs: {jobs: {terms: {field:job.keyword},aggs:{old_employee:{top_hits:{size:3,sort:[{age:{order:desc}}]}}}}} } 运行结果 {took : 13,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7,old_employee : {hits : {total : {value : 7,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 11,_score : null,_source : {name : Jenny,age : 36,job : Java Programmer,gender : female,salary : 38000},sort : [36]},{_index : employees,_type : _doc,_id : 15,_score : null,_source : {name : King,age : 33,job : Java Programmer,gender : male,salary : 28000},sort : [33]},{_index : employees,_type : _doc,_id : 9,_score : null,_source : {name : Gregory,age : 32,job : Java Programmer,gender : male,salary : 22000},sort : [32]}]}}},{key : Javascript Programmer,doc_count : 4,old_employee : {hits : {total : {value : 4,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 14,_score : null,_source : {name : Marshall,age : 32,job : Javascript Programmer,gender : male,salary : 25000},sort : [32]},{_index : employees,_type : _doc,_id : 18,_score : null,_source : {name : Catherine,age : 29,job : Javascript Programmer,gender : female,salary : 20000},sort : [29]},{_index : employees,_type : _doc,_id : 17,_score : null,_source : {name : Goodwin,age : 25,job : Javascript Programmer,gender : male,salary : 16000},sort : [25]}]}}},{key : QA,doc_count : 3,old_employee : {hits : {total : {value : 3,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 6,_score : null,_source : {name : Lucy,age : 31,job : QA,gender : female,salary : 25000},sort : [31]},{_index : employees,_type : _doc,_id : 7,_score : null,_source : {name : Byrd,age : 27,job : QA,gender : male,salary : 20000},sort : [27]},{_index : employees,_type : _doc,_id : 5,_score : null,_source : {name : Rose,age : 25,job : QA,gender : female,salary : 18000},sort : [25]}]}}},{key : DBA,doc_count : 2,old_employee : {hits : {total : {value : 2,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 19,_score : null,_source : {name : Boone,age : 30,job : DBA,gender : male,salary : 30000},sort : [30]},{_index : employees,_type : _doc,_id : 20,_score : null,_source : {name : Kathy,age : 29,job : DBA,gender : female,salary : 20000},sort : [29]}]}}},{key : Web Designer,doc_count : 2,old_employee : {hits : {total : {value : 2,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 4,_score : null,_source : {name : Rivera,age : 26,job : Web Designer,gender : female,salary : 22000},sort : [26]},{_index : employees,_type : _doc,_id : 3,_score : null,_source : {name : Tran,age : 25,job : Web Designer,gender : male,salary : 18000},sort : [25]}]}}},{key : Dev Manager,doc_count : 1,old_employee : {hits : {total : {value : 1,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 2,_score : null,_source : {name : Underwood,age : 41,job : Dev Manager,gender : male,salary : 50000},sort : [41]}]}}},{key : Product Manager,doc_count : 1,old_employee : {hits : {total : {value : 1,relation : eq},max_score : null,hits : [{_index : employees,_type : _doc,_id : 1,_score : null,_source : {name : Emma,age : 32,job : Product Manager,gender : female,salary : 35000},sort : [32]}]}}}]}} } 嵌套聚合示例 # 嵌套聚合1按照工作类型分桶并统计工资信息 POST employees/_search {size: 0,aggs: {Job_salary_stats: {terms: {field: job.keyword},aggs: {salary: {stats: {field: salary}}}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {Job_salary_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7,salary : {count : 7,min : 9000.0,max : 38000.0,avg : 25571.428571428572,sum : 179000.0}},{key : Javascript Programmer,doc_count : 4,salary : {count : 4,min : 16000.0,max : 25000.0,avg : 19250.0,sum : 77000.0}},{key : QA,doc_count : 3,salary : {count : 3,min : 18000.0,max : 25000.0,avg : 21000.0,sum : 63000.0}},{key : DBA,doc_count : 2,salary : {count : 2,min : 20000.0,max : 30000.0,avg : 25000.0,sum : 50000.0}},{key : Web Designer,doc_count : 2,salary : {count : 2,min : 18000.0,max : 22000.0,avg : 20000.0,sum : 40000.0}},{key : Dev Manager,doc_count : 1,salary : {count : 1,min : 50000.0,max : 50000.0,avg : 50000.0,sum : 50000.0}},{key : Product Manager,doc_count : 1,salary : {count : 1,min : 35000.0,max : 35000.0,avg : 35000.0,sum : 35000.0}}]}} } # 多次嵌套。根据工作类型分桶然后按照性别分桶计算工资的统计信息 POST employees/_search {size: 0,aggs: {Job_gender_stats: {terms: {field: job.keyword},aggs: {gender_stats: {terms: {field: gender},aggs: {salary_stats: {stats: {field: salary}}}}}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {Job_gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : male,doc_count : 5,salary_stats : {count : 5,min : 9000.0,max : 32000.0,avg : 22200.0,sum : 111000.0}},{key : female,doc_count : 2,salary_stats : {count : 2,min : 30000.0,max : 38000.0,avg : 34000.0,sum : 68000.0}}]}},{key : Javascript Programmer,doc_count : 4,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : male,doc_count : 3,salary_stats : {count : 3,min : 16000.0,max : 25000.0,avg : 19000.0,sum : 57000.0}},{key : female,doc_count : 1,salary_stats : {count : 1,min : 20000.0,max : 20000.0,avg : 20000.0,sum : 20000.0}}]}},{key : QA,doc_count : 3,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : female,doc_count : 2,salary_stats : {count : 2,min : 18000.0,max : 25000.0,avg : 21500.0,sum : 43000.0}},{key : male,doc_count : 1,salary_stats : {count : 1,min : 20000.0,max : 20000.0,avg : 20000.0,sum : 20000.0}}]}},{key : DBA,doc_count : 2,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : female,doc_count : 1,salary_stats : {count : 1,min : 20000.0,max : 20000.0,avg : 20000.0,sum : 20000.0}},{key : male,doc_count : 1,salary_stats : {count : 1,min : 30000.0,max : 30000.0,avg : 30000.0,sum : 30000.0}}]}},{key : Web Designer,doc_count : 2,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : female,doc_count : 1,salary_stats : {count : 1,min : 22000.0,max : 22000.0,avg : 22000.0,sum : 22000.0}},{key : male,doc_count : 1,salary_stats : {count : 1,min : 18000.0,max : 18000.0,avg : 18000.0,sum : 18000.0}}]}},{key : Dev Manager,doc_count : 1,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : male,doc_count : 1,salary_stats : {count : 1,min : 50000.0,max : 50000.0,avg : 50000.0,sum : 50000.0}}]}},{key : Product Manager,doc_count : 1,gender_stats : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : female,doc_count : 1,salary_stats : {count : 1,min : 35000.0,max : 35000.0,avg : 35000.0,sum : 35000.0}}]}}]}} } 4.4 Pipeline Aggregation 支持对聚合分析的结果再次进行聚合分析。 Pipeline 的分析结果会输出到原结果中根据位置的不同分为两类 Sibling - 结果和现有分析结果同级 MaxminAvg Sum BucketStatsExtended Status BucketPercentiles BucketParent -结果内嵌到现有的聚合分析结果之中 Derivative(求导)Cumultive Sum(累计求和)Moving Function(移动平均值 ) min_bucket示例 在员工数最多的工种里找出平均工资最低的工种 # 平均工资最低的工种 POST employees/_search {size: 0,aggs: {jobs: {terms: {field: job.keyword,size: 10},aggs: {avg_salary: {avg: {field: salary}}}},min_salary_by_job:{ min_bucket: { buckets_path: jobsavg_salary }}} } 运行结果 {took : 2,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7,avg_salary : {value : 25571.428571428572}},{key : Javascript Programmer,doc_count : 4,avg_salary : {value : 19250.0}},{key : QA,doc_count : 3,avg_salary : {value : 21000.0}},{key : DBA,doc_count : 2,avg_salary : {value : 25000.0}},{key : Web Designer,doc_count : 2,avg_salary : {value : 20000.0}},{key : Dev Manager,doc_count : 1,avg_salary : {value : 50000.0}},{key : Product Manager,doc_count : 1,avg_salary : {value : 35000.0}}]},min_salary_by_job : {value : 19250.0,keys : [Javascript Programmer]}} } min_salary_by_job结果和jobs的聚合同级min_bucket求之前结果的最小值通过bucket_path关键字指定路径 Stats示例 # 平均工资的统计分析 POST employees/_search {size: 0,aggs: {jobs: {terms: {field: job.keyword,size: 10},aggs: {avg_salary: {avg: {field: salary}}}},stats_salary_by_job:{stats_bucket: {buckets_path: jobsavg_salary}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7,avg_salary : {value : 25571.428571428572}},{key : Javascript Programmer,doc_count : 4,avg_salary : {value : 19250.0}},{key : QA,doc_count : 3,avg_salary : {value : 21000.0}},{key : DBA,doc_count : 2,avg_salary : {value : 25000.0}},{key : Web Designer,doc_count : 2,avg_salary : {value : 20000.0}},{key : Dev Manager,doc_count : 1,avg_salary : {value : 50000.0}},{key : Product Manager,doc_count : 1,avg_salary : {value : 35000.0}}]},stats_salary_by_job : {count : 7,min : 19250.0,max : 50000.0,avg : 27974.48979591837,sum : 195821.42857142858}} } percentiles示例 # 平均工资的百分位数 POST employees/_search {size: 0,aggs: {jobs: {terms: {field: job.keyword,size: 10},aggs: {avg_salary: {avg: {field: salary}}}},percentiles_salary_by_job:{percentiles_bucket: {buckets_path: jobsavg_salary}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7,avg_salary : {value : 25571.428571428572}},{key : Javascript Programmer,doc_count : 4,avg_salary : {value : 19250.0}},{key : QA,doc_count : 3,avg_salary : {value : 21000.0}},{key : DBA,doc_count : 2,avg_salary : {value : 25000.0}},{key : Web Designer,doc_count : 2,avg_salary : {value : 20000.0}},{key : Dev Manager,doc_count : 1,avg_salary : {value : 50000.0}},{key : Product Manager,doc_count : 1,avg_salary : {value : 35000.0}}]},percentiles_salary_by_job : {values : {1.0 : 19250.0,5.0 : 19250.0,25.0 : 21000.0,50.0 : 25000.0,75.0 : 35000.0,95.0 : 50000.0,99.0 : 50000.0}}} } Cumulative_sum示例 #Cumulative_sum 累计求和 POST employees/_search {size: 0,aggs: {age: {histogram: {field: age,min_doc_count: 0,interval: 1},aggs: {avg_salary: {avg: {field: salary}},cumulative_salary:{cumulative_sum: {buckets_path: avg_salary}}}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {age : {buckets : [{key : 20.0,doc_count : 1,avg_salary : {value : 9000.0},cumulative_salary : {value : 9000.0}},{key : 21.0,doc_count : 1,avg_salary : {value : 16000.0},cumulative_salary : {value : 25000.0}},{key : 22.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 25000.0}},{key : 23.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 25000.0}},{key : 24.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 25000.0}},{key : 25.0,doc_count : 3,avg_salary : {value : 17333.333333333332},cumulative_salary : {value : 42333.33333333333}},{key : 26.0,doc_count : 1,avg_salary : {value : 22000.0},cumulative_salary : {value : 64333.33333333333}},{key : 27.0,doc_count : 2,avg_salary : {value : 20000.0},cumulative_salary : {value : 84333.33333333333}},{key : 28.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 84333.33333333333}},{key : 29.0,doc_count : 2,avg_salary : {value : 20000.0},cumulative_salary : {value : 104333.33333333333}},{key : 30.0,doc_count : 2,avg_salary : {value : 30000.0},cumulative_salary : {value : 134333.3333333333}},{key : 31.0,doc_count : 2,avg_salary : {value : 28500.0},cumulative_salary : {value : 162833.3333333333}},{key : 32.0,doc_count : 3,avg_salary : {value : 27333.333333333332},cumulative_salary : {value : 190166.66666666666}},{key : 33.0,doc_count : 1,avg_salary : {value : 28000.0},cumulative_salary : {value : 218166.66666666666}},{key : 34.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 218166.66666666666}},{key : 35.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 218166.66666666666}},{key : 36.0,doc_count : 1,avg_salary : {value : 38000.0},cumulative_salary : {value : 256166.66666666666}},{key : 37.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 256166.66666666666}},{key : 38.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 256166.66666666666}},{key : 39.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 256166.66666666666}},{key : 40.0,doc_count : 0,avg_salary : {value : null},cumulative_salary : {value : 256166.66666666666}},{key : 41.0,doc_count : 1,avg_salary : {value : 50000.0},cumulative_salary : {value : 306166.6666666666}}]}} } 4.5 聚合的作用范围 ES聚合分析的默认作用范围是query的查询结果集同时ES还支持以下方式改变聚合的作用范围 FilterPost FilterGlobal #Query POST employees/_search {size: 0,query: {range: {age: {gte: 20}}},aggs: {jobs: {terms: {field:job.keyword}}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 10,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 5},{key : DBA,doc_count : 1},{key : Dev Manager,doc_count : 1},{key : Javascript Programmer,doc_count : 1},{key : Product Manager,doc_count : 1},{key : QA,doc_count : 1}]}} } #Filter POST employees/_search {size: 0,aggs: {older_person: {filter:{range:{age:{from:35}}},aggs:{jobs:{terms: {field:job.keyword}}}},all_jobs: {terms: {field:job.keyword}}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {older_person : {doc_count : 2,jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Dev Manager,doc_count : 1},{key : Java Programmer,doc_count : 1}]}},all_jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7},{key : Javascript Programmer,doc_count : 4},{key : QA,doc_count : 3},{key : DBA,doc_count : 2},{key : Web Designer,doc_count : 2},{key : Dev Manager,doc_count : 1},{key : Product Manager,doc_count : 1}]}} } #Post field. 一条语句找出所有的job类型。还能找到聚合后符合条件的结果 POST employees/_search {aggs: {jobs: {terms: {field: job.keyword}}},post_filter: {match: {job.keyword: Dev Manager}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 1,relation : eq},max_score : 1.0,hits : [{_index : employees,_type : _doc,_id : 2,_score : 1.0,_source : {name : Underwood,age : 41,job : Dev Manager,gender : male,salary : 50000}}]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Java Programmer,doc_count : 7},{key : Javascript Programmer,doc_count : 4},{key : QA,doc_count : 3},{key : DBA,doc_count : 2},{key : Web Designer,doc_count : 2},{key : Dev Manager,doc_count : 1},{key : Product Manager,doc_count : 1}]}} } #global POST employees/_search {size: 0,query: {range: {age: {gte: 40}}},aggs: {jobs: {terms: {field:job.keyword}},all:{global:{},aggs:{salary_avg:{avg:{field:salary}}}}} } 运行结果 {took : 0,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 1,relation : eq},max_score : null,hits : [ ]},aggregations : {all : {doc_count : 20,salary_avg : {value : 24700.0}},jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Dev Manager,doc_count : 1}]}} } 4.6 排序 指定order按照count和key进行排序 默认情况按照count降序排序指定size就能返回相应的桶 #排序 order #count and key POST employees/_search {size: 0,query: {range: {age: {gte: 20}}},aggs: {jobs: {terms: {field:job.keyword,order:[{_count:asc},{_key:desc}]}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Product Manager,doc_count : 1},{key : Dev Manager,doc_count : 1},{key : Web Designer,doc_count : 2},{key : DBA,doc_count : 2},{key : QA,doc_count : 3},{key : Javascript Programmer,doc_count : 4},{key : Java Programmer,doc_count : 7}]}} } #排序 order #count and key POST employees/_search {size: 0,aggs: {jobs: {terms: {field:job.keyword,order:[ {avg_salary:desc}]},aggs: {avg_salary: {avg: {field:salary}}}}} } 运行结果 {took : 2,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Dev Manager,doc_count : 1,avg_salary : {value : 50000.0}},{key : Product Manager,doc_count : 1,avg_salary : {value : 35000.0}},{key : Java Programmer,doc_count : 7,avg_salary : {value : 25571.428571428572}},{key : DBA,doc_count : 2,avg_salary : {value : 25000.0}},{key : QA,doc_count : 3,avg_salary : {value : 21000.0}},{key : Web Designer,doc_count : 2,avg_salary : {value : 20000.0}},{key : Javascript Programmer,doc_count : 4,avg_salary : {value : 19250.0}}]}} } #排序 order #count and key POST employees/_search {size: 0,aggs: {jobs: {terms: {field:job.keyword,order:[ {stats_salary.min:desc}]},aggs: {stats_salary: {stats: {field:salary}}}}} } 运行结果 {took : 1,timed_out : false,_shards : {total : 1,successful : 1,skipped : 0,failed : 0},hits : {total : {value : 20,relation : eq},max_score : null,hits : [ ]},aggregations : {jobs : {doc_count_error_upper_bound : 0,sum_other_doc_count : 0,buckets : [{key : Dev Manager,doc_count : 1,stats_salary : {count : 1,min : 50000.0,max : 50000.0,avg : 50000.0,sum : 50000.0}},{key : Product Manager,doc_count : 1,stats_salary : {count : 1,min : 35000.0,max : 35000.0,avg : 35000.0,sum : 35000.0}},{key : DBA,doc_count : 2,stats_salary : {count : 2,min : 20000.0,max : 30000.0,avg : 25000.0,sum : 50000.0}},{key : QA,doc_count : 3,stats_salary : {count : 3,min : 18000.0,max : 25000.0,avg : 21000.0,sum : 63000.0}},{key : Web Designer,doc_count : 2,stats_salary : {count : 2,min : 18000.0,max : 22000.0,avg : 20000.0,sum : 40000.0}},{key : Javascript Programmer,doc_count : 4,stats_salary : {count : 4,min : 16000.0,max : 25000.0,avg : 19250.0,sum : 77000.0}},{key : Java Programmer,doc_count : 7,stats_salary : {count : 7,min : 9000.0,max : 38000.0,avg : 25571.428571428572,sum : 179000.0}}]}} } 4.7 ES聚合分析不精准原因分析 ElasticSearch在对海量数据进行聚合分析的时候会损失搜索的精准度来满足实时性的需求。 Terms聚合分析的执行流程 不精准的原因 数据分散到多个分片聚合是每个分片的取 Top X导致结果不精准。ES 可以不每个分片Top X而是全量聚合但势必这会有很大的性能问题。 思考如何提高聚合精确度 方案1设置主分片为1 注意7.x版本已经默认为1。 适用场景数据量小的小集群规模业务场景。 方案2调大 shard_size 值 设置 shard_size 为比较大的值官方推荐size*1.510。shard_size 值越大结果越趋近于精准聚合结果值。此外还可以通过show_term_doc_count_error参数显示最差情况下的错误值用于辅助确定 shard_size 大小。 size是聚合结果的返回值客户期望返回聚合排名前三size值就是 3。shard_size: 每个分片上聚合的数据条数。shard_size 原则上要大于等于 size 适用场景数据量大、分片数多的集群业务场景。 测试 使用kibana的测试数据 DELETE my_flights PUT my_flights {settings: {number_of_shards: 20},mappings : {properties : {AvgTicketPrice : {type : float},Cancelled : {type : boolean},Carrier : {type : keyword},Dest : {type : keyword},DestAirportID : {type : keyword},DestCityName : {type : keyword},DestCountry : {type : keyword},DestLocation : {type : geo_point},DestRegion : {type : keyword},DestWeather : {type : keyword},DistanceKilometers : {type : float},DistanceMiles : {type : float},FlightDelay : {type : boolean},FlightDelayMin : {type : integer},FlightDelayType : {type : keyword},FlightNum : {type : keyword},FlightTimeHour : {type : keyword},FlightTimeMin : {type : float},Origin : {type : keyword},OriginAirportID : {type : keyword},OriginCityName : {type : keyword},OriginCountry : {type : keyword},OriginLocation : {type : geo_point},OriginRegion : {type : keyword},OriginWeather : {type : keyword},dayOfWeek : {type : integer},timestamp : {type : date}}} }POST _reindex {source: {index: kibana_sample_data_flights},dest: {index: my_flights} }GET my_flights/_count GET kibana_sample_data_flights/_search {size: 0,aggs: {weather: {terms: {field:OriginWeather,size:5,show_term_doc_count_error:true}}} }GET my_flights/_search {size: 0,aggs: {weather: {terms: {field:OriginWeather,size:5,shard_size:10,show_term_doc_count_error:true}}} } 在Terms Aggregation的返回中有两个特殊的数值 doc_count_error_upper_bound : 被遗漏的term 分桶包含的文档有可能的最大值sum_other_doc_count: 除了返回结果 bucket的terms以外其他 terms 的文档总数总数-返回的总数) 方案3将size设置为全量值来解决精度问题 将size设置为2的32次方减去1也就是分片支持的最大值来解决精度问题。 原因1.x版本size等于 0 代表全部高版本取消 0 值所以设置了最大值大于业务的全量值。 全量带来的弊端就是如果分片数据量极大这样做会耗费巨大的CPU 资源来排序而且可能会阻塞网络。 适用场景对聚合精准度要求极高的业务场景由于性能问题不推荐使用。 方案4使用Clickhouse/ Spark 进行精准聚合 适用场景数据量非常大、聚合精度要求高、响应速度快的业务场景。 4.8 Elasticsearch 聚合性能优化 启用 eager global ordinals 提升高基数聚合性能 适用场景高基数聚合 。高基数聚合场景中的高基数含义一个字段包含很大比例的唯一值。 global ordinals 中文翻译成全局序号是一种数据结构应用场景如下 基于 keywordip 等字段的分桶聚合包含terms聚合、composite 聚合等。基于text 字段的分桶聚合前提条件是fielddata 开启。基于父子文档 Join 类型的 has_child 查询和 父聚合。 global ordinals 使用一个数值代表字段中的字符串值然后为每一个数值分配一个 bucket分桶。 global ordinals 的本质是启用 eager_global_ordinals 时会在刷新refresh分片时构建全局序号。这将构建全局序号的成本从搜索阶段转移到了数据索引化写入阶段。 创建索引的同时开启eager_global_ordinals。 PUT /my-index {mappings: {properties: {tags: {type: keyword,eager_global_ordinals: true}}} 注意开启 eager_global_ordinals 会影响写入性能因为每次刷新时都会创建新的全局序号。为了最大程度地减少由于频繁刷新建立全局序号而导致的额外开销请调大刷新间隔 refresh_interval。 动态调整刷新频率的方法如下 PUT my-index/_settings {index: {refresh_interval: 30s} 该招数的本质是以空间换时间。 插入数据时对索引进行预排序 Index sorting 索引排序可用于在插入时对索引进行预排序而不是在查询时再对索引进行排序这将提高范围查询range query和排序操作的性能。在 Elasticsearch 中创建新索引时可以配置如何对每个分片内的段进行排序。这是 Elasticsearch 6.X 之后版本才有的特性。 PUT /my_index {settings: {index:{sort.field: create_time,sort.order: desc}},mappings: {properties: {create_time:{type: date}}} } 注意预排序将增加 Elasticsearch 写入的成本。在某些用户特定场景下开启索引预排序会导致大约 40%-50% 的写性能下降。也就是说如果用户场景更关注写性能的业务开启索引预排序不是一个很好的选择。 使用节点查询缓存 节点查询缓存Node query cache可用于有效缓存过滤器filter操作的结果。如果多次执行同一 filter 操作这将很有效但是即便更改过滤器中的某一个值也将意味着需要计算新的过滤器结果。 例如由于 “now” 值一直在变化因此无法缓存在过滤器上下文中使用 “now” 的查询。 那怎么使用缓存呢通过在 now 字段上应用 datemath 格式将其四舍五入到最接近的分钟/小时等可以使此类请求更具可缓存性以便可以对筛选结果进行缓存。 PUT /my_index/_doc/1 {create_time:2022-05-11T16:30:55.328Z }#下面的示例无法使用缓存 GET /my_index/_search {query:{constant_score: {filter: {range: {create_time: {gte: now-1h,lte: now}}}}} }# 下面的示例就可以使用节点查询缓存。 GET /my_index/_search {query:{constant_score: {filter: {range: {create_time: {gte: now-1h/m,lte: now/m}}}}} } 上述示例中的“now-1h/m” 就是 datemath 的格式。 如果当前时间 now 是16:31:29那么range query 将匹配 my_date 介于15:31:00 和 15:31:59 之间的时间数据。同理聚合的前半部分 query 中如果有基于时间查询或者后半部分 aggs 部分中有基于时间聚合的建议都使用 datemath 方式做缓存处理以优化性能。 使用分片请求缓存 聚合语句中设置size0就会使用分片请求缓存缓存结果。size 0 的含义是只返回聚合结果不返回查询结果。 GET /es_db/_search {size: 0,aggs: {remark_agg: {terms: {field: remark.keyword}}} } 拆分聚合使聚合并行化 Elasticsearch 查询条件中同时有多个条件聚合默认情况下聚合不是并行运行的。当为每个聚合提供自己的查询并执行 msearch 时性能会有显著提升。因此在 CPU 资源不是瓶颈的前提下如果想缩短响应时间可以将多个聚合拆分为多个查询借助msearch 实现并行聚合。 #常规的多条件聚合实现 GET /employees/_search {size: 0,aggs: {job_agg: {terms: {field: job.keyword}},max_salary:{max: {field: salary}}} } # msearch 拆分多个语句的聚合实现 GET _msearch {index:employees} {size:0,aggs:{job_agg:{terms:{field: job.keyword}}}} {index:employees} {size:0,aggs:{max_salary:{max:{field: salary}}}}
http://www.laogonggong.com/news/122001.html

相关文章:

  • 兴宁网站设计河南优化公司
  • 长沙网站优化方式公司建设网站费用直接列支
  • 酒店网站程序宁波seo排名费用
  • 网站开发对企业有什么用h5是什么意思游戏
  • 怎么做自己网站里的资讯建外做网站的公司
  • 站长全网指数查询佛山seo网站
  • 有教做衣服的网站吗做网站开发学什么语言
  • 岳阳网站定制手机上做网站
  • 小门户网站开发c++可视化界面设计
  • 抚州专业的企业网站建设公司百度推广和百度竞价有什么区别
  • 西安做百度网站公司律师事务所网站方案
  • 广东专业网站定制软件开发学什么专业好
  • wordpress stop国外网站谷歌seo推广
  • 一条龙建站多少钱如何注册一个域名
  • 网站开发的前后台的步骤分别为手机网站在哪里找到
  • 网站整合方案wordpress 前台 插件
  • 济南外贸网站制作专门做淘宝代运营的网站
  • 营销网站运营的基本环节计算机软件著作权
  • 浙江网站建设与维护书网站公司必须帮备案
  • 建设移动网站shopify是什么平台
  • 深圳市房屋管理局官方网站手机优化大师下载安装
  • 周口seo 网站网站登录到wordpress
  • 晚上必看的正能量网站app项链seo关键词
  • 银川做网站公司唐河企业网站制作哪家好
  • 网站开发敲代码wordpress 页面颜色
  • 苏宁推客如何做网站长沙模板建站定制
  • 免费建站工具有哪些重庆人才招聘网最新招聘2022
  • 如何建企业仢网站google官方入口
  • 网站推广平台怎么做三亚网上商城
  • 1元购网站建设网络平台运营是做什么的