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

襄汾县住房和建设局网站中学生网站源码

襄汾县住房和建设局网站,中学生网站源码,做响应式网站哪家公司好,diywap手机微网站内容管理系统文章目录一、ElasticSearch概述1、ElasticSearch是什么2、全文搜索引擎3、ElasticSearch 和 Solr3.1 概述3.2 比较总结二、Elasticsearch入门1、Elasticsearch安装1.1 下载使用1.2 数据格式2、索引操作3、文档操作#xff08;了解#xff09;3.1 创建文档3.2 文档查询3.3 文档… 文章目录一、ElasticSearch概述1、ElasticSearch是什么2、全文搜索引擎3、ElasticSearch 和 Solr3.1 概述3.2 比较总结二、Elasticsearch入门1、Elasticsearch安装1.1 下载使用1.2 数据格式2、索引操作3、文档操作了解3.1 创建文档3.2 文档查询3.3 文档修改3.4 文档删除3.5 文档条件查询 分页查询 查询排序3.6 文档多条件查询 范围查询3.7 文档全文检索 完全匹配 高亮查询3.8 文档聚合查询3.9 文档映射关系4、JavaAPI4.1 环境准备4.2 索引-创建4.3 索引查询 删除4.4 文档新增 修改4.5 文档-批量查询 批量删除4.6 文档-高级查询-全量4.7 文档-高级查询-termQuery查询、分页查询、排序查询、过滤字段4.8 文档-高级查询-组合查询 and、or、范围查询4.9 文档-高级查询-模糊、高亮查询4.10 文档-聚合查询三、ElasticSearch 环境1、相关概念1.1 单机 集群1.2 集群Cluster1.3 节点 Node2、Windows集群3、Linux单机4、Linux集群四、Elasticsearch进阶1、核心概念1.1 索引index1.2 类型 type1.3 文档Document1.4 字段Field1.5 映射Mapping1.6 分片Shards1.7 副本Replicas1.8 分配Allocation2、系统架构3、分布式集群3.1 单节点集群3.2 故障转移3.3 水平扩容3.4 应对故障3.5 路由计算 分片控制3.6 数据写流程3.7 数据读流程3.8 更新流程 批量操作流程4、分片原理4.1 倒排索引4.2 文档搜索4.3 动态更新索引4.4 文档刷新 文档刷写 文档合并近实时搜索持久化变更段合并4.5 文档分析内置分析器分析器使用场景测试分析器指定分析器IK分词器自定义分析器4.6 文档控制文档冲突乐观并发控制外部系统版本控制4.7 文档展示-Kibana五、Elasticsearch 集成1、SpringData 框架集成1.1 Spring Data 框架使用1.2 Spring Data Elasticsearch 介绍1.3 框架集成1.4 集成测试-索引操作1.5 集成测试-文档操作1.6 集成测试-文档搜索2、Spark Streaming 框架集成2.1 Spark Streaming 框架介绍2.2 框架集成3、Flink 框架集成3.1 Flink 框架介绍3.2 demo实现六、Elasticsearch 优化1、硬件选择2、分片策略2.1 合理设置分片数2.2 推迟分片分配3、路由选择4、写入速度优化4.1 优化存储设备4.2 合理使用合并4.3 减少 Refresh 的次数4.4 加大 Flush 设置4.5 减少副本的数量5、内存设置6、重要配置项七、Elasticsearch面试题1、为什么要使用 Elasticsearch2、Elasticsearch 的 master 选举流程3、Elasticsearch 集群脑裂问题4、Elasticsearch 索引文档的流程5、Elasticsearch 更新和删除文档的流程6、Elasticsearch 搜索的流程7、Elasticsearch 在部署时对 Linux 的设置有哪些优化方法8、GC 方面在使用 Elasticsearch 时要注意什么9、Elasticsearch 对于大数据量上亿量级的聚合如何实现10、在并发情况下 Elasticsearch 如果保证读写一致11、如何监控 Elasticsearch 集群状态12、是否了解字典树13、Elasticsearch 中的集群、节点、索引、文档、类型是什么14、Elasticsearch 中的倒排索引是什么一、ElasticSearch概述 1、ElasticSearch是什么 Elasticsearch是一个实时的分布式搜索和分析引擎。它可以帮助你用前所未有的速度去处理大规模数据。它可以用于全文搜索结构化搜索以及分析当然你也可以将这三者进行组合。 Elasticsearch是一个建立在全文搜索引擎 Apache Lucene™ 基础上的搜索引擎可以说Lucene是当今最先进最高效的全功能开源搜索引擎框架。但是Lucene只是一个框架要充分利用它的功能需要使用JAVA并且在程序中集成Lucene。当然Elasticsearch并不仅仅是Lucene这么简单它不但包括了全文搜索功能还可以进行以下工作: 分布式实时文件存储并将每一个字段都编入索引使其可以被搜索。实时分析的分布式搜索引擎。可以扩展到上百台服务器处理PB级别的结构化或非结构化数据。 2、全文搜索引擎 Google百度类的网站搜索它们都是根据网页中的关键字生成索引我们在搜索的时候输入关键字它们会将该关键字即索引匹配到的所有网页返回还有常见的项目中应用日志的搜索等等。对于这些非结构化的数据文本关系型数据库搜索不是能很好的支持。 一般传统数据库全文检索都实现的很鸡肋因为一般也没人用数据库存文本字段。进行全文检索需要扫描整个表如果数据量大的话即使对 SQL 的语法优化也收效甚微。建立了索引但是维护起来也很麻烦对于 insert 和 update 操作都会重新构建索引。 基于以上原因可以分析得出在一些生产环境中使用常规的搜索方式性能是非常差的 搜索的数据对象是大量的非结构化的文本数据文件记录量达到数十万或数百万个甚至更多支持大量基于交互式文本的查询需求非常灵活的全文搜索查询对高度相关的搜索结果的有特殊需求但是没有可用的关系数据库可以满足对不同记录类型、非文本数据操作或安全事务处理的需求相对较少的情况。为了解决结构化数据搜索和非结构化数据搜索性能问题我们就需要专业健壮强大的全文搜索引擎 这里说到的全文搜索引擎指的是目前广泛应用的主流搜索引擎。它的工作原理是计算机索引程序通过扫描文章中的每一个词对每一个词建立一个索引指明该词在文章中出现的次数和位置当用户查询时检索程序就根据事先建立的索引进行查找并将查找的结果反馈给用户的检索方式这个过程类似于通过字典中的检索字表查字的过程 3、ElasticSearch 和 Solr 3.1 概述 Lucene是apache软件基金会Jakarta项目组的一个子项目提供了一个简单却强大的应用程序式接口能够做全文索引和搜寻。在Java开发环境里Lucene是一个成熟的免费开源工具。就其本身而言Lucene是当前提供全文搜索功能类库的核心工具包而真正使用它还需要一个完善服务框架搭建起来进行应用。 目前市面上流行的搜索引擎软件主流的就两款Elasticsearch和Solr,这两款都是基于Lucene搭建的可以独立部署启动的搜索引擎服务软件。由于内核相同所以两者除了服务器安装、部署、管理、集群外对于数据的操作 修改、添加、保存、查询等等都十分相似。 在使用过程中一般都会将Elastics search和Solr这两个软件对比然后进行选型。这两个搜索引擎都是流行的先进的的开源搜索引擎它们都是围绕核心底层搜索库-Lucene构建的但它们又是不同的。像所有东西并样每个都有其优点和缺点 当单纯的对已有数据进行搜索时Solr更快当实时建立索引时, Solr会产生io阻塞查询性能较差, Elasticsearch具有明显的优势随着数据量的增加Solr的搜索效率会变得更低而Elasticsearch却没有明显的变化 综上所述Solr的架构不适合实时搜索的应用 3.2 比较总结 Solr 利用 Zookeeper 进行分布式管理而 Elasticsearch 自身带有分布式协调管理功能Solr 支持更多格式的数据而 Elasticsearch 仅支持json文件格式Solr 官方提供的功能更多而 Elasticsearch 本身更注重于核心功能高级功能多有第三方插件提供Solr 在传统的搜索应用中表现好于 Elasticsearch但在处理实时搜索应用时效率明显低于 Elasticsearch Solr 是传统搜索应用的有力解决方案但 Elasticsearch 更适用于新兴的实时搜索应用。 对于两者的使用 由于易于使用 Elasticsearch 在新开发者中更受欢迎。一个下载和一个命令就可以启动一切如果除了搜索文本之外还需要它来处理分析查询 Elasticsearch 是更好的选择如果需要分布式索引则需要选择 Elasticsearch 。对于需要良好可伸缩性和以及性能分布式环境 Elasticsearch 是更好的选择lasticsearch 在开源日志管理用例中占据主导地位许多组织在 Elasticsearch 中索引它们的日志以使其可搜索如果你喜欢监控和指标那么请使用Elastics search因为相对于Solr Elasticsearch 暴露了更多的关键指标 二、Elasticsearch入门 1、Elasticsearch安装 1.1 下载使用 官网地址https://www.elastic.co/cn/ Elasticsearch 7.8.0下载界面https://www.elastic.co/cn/downloads/past-releases/elasticsearch-7-8-0 Windows 版的 Elasticsearch 压缩包解压即安装完毕解压后的 Elasticsearch 的目录结构如下 目录含义bin可执行脚本目录config配置目录jdk内置 JDK 目录lib类库logs日志目录modules模块目录plugins插件目录 解压后进入 bin 文件目录点击 elasticsearch.bat 文件启动 ES 服务 。注意 9300 端口为 Elasticsearch 集群间组件的通信端口 9200 端口为浏览器访问的 http协议 RESTful 端口。打开浏览器输入地址 http://localhost:9200测试返回结果 因为ES使用的是RESTful JSON传输数据所以需要一个发送http请求的客户端推荐Postman或者Apifox 1.2 数据格式 Elassitiesearch 是面向文档型数据库一条数据在这里就是一个文档。为了方便大家理解我们将 Elastlesearch 里存储文档数据和关系型数据库MySQL存储数据的概念进行一个类比 ES 里的 Index 可以看做一个库而 Types 相当于表 Documents 则相当于表的行。这里 Types 的概念已经被逐渐弱化 Elasticsearch 6.X 中一个 index 下已经只能包含一个type Elasticsearch 7.X 中, Type 的概念已经被删除了。 ES还包括了倒排索引这个极大方便了数据的搜索 正排索引传统 idcontent1001my name is zhang san1002my name is li si 倒排索引 keywordidname1001, 1002zhang1001 2、索引操作 # 对比关系型数据块创建索引就等同于创建数据库需要发送put请求 http://127.0.0.1:9200/shopping # 查看单个索引信息发送get http://127.0.0.1:9200/shopping # 查看所有索引信息,发送get http://127.0.0.1:9200/_cat/indices?v # 删除索引信息,向 ES 服务器发 DELETE 请求 http://127.0.0.1:9200/shopping 3、文档操作了解 3.1 创建文档 假设索引已经创建好了接下来我们来创建文档并添加数据。这里的文档可以类比为关系型数据库中的表数据添加的数据格式为 JSON 格式向 ES 服务器发 **POST **请求 http://127.0.0.1:9200/shopping/_doc注意此处发送请求的方式必须为 POST不能是 PUT否则会发生错误 {title:小米手机,category:小米,images:http://www.xxx.com/xm.jpg,price:3999.00 }上面的数据创建后由于没有指定数据唯一性标识ID默认情况下 ES 服务器会随机生成一个。如果想要自定义唯一性标识需要在创建时指定 http://127.0.0.1:9200/shopping/_doc/1001 此处需要注意如果增加数据时明确数据主键那么请求方式也可以为 PUT其中路径**_doc**可以改为_create也没有问题(符合幂等性) 3.2 文档查询 根据主键查询向 ES 服务器发 GET请求 http://127.0.0.1:9200/shopping/_doc/1001查询所有信息向 ES 服务器发 GET请求 http://127.0.0.1:9200/shopping/_search 3.3 文档修改 全量修改和新增文档一样输入相同的 URL 地址请求如果请求体变化会将原有的数据内容覆盖向 ES 服务器发 PUT请求 http://127.0.0.1:9200/shopping/_doc/1。提示全量更新每次结果都是相同的所以是幂等性的使用PUT局部修改向 ES 服务器发 POST请求 http://127.0.0.1:9200/shopping/_update/1001{doc: {title:华为手机} } 3.4 文档删除 向 ES 服务器发 DELETE 请求 http://127.0.0.1:9200/shopping/_doc/1001 3.5 文档条件查询 分页查询 查询排序 # 查找category为小米的文档向 ES 发GET请求 http://127.0.0.1:9200/shopping/_search?qcategory:小米 # 上述不常用一般使用带JSON请求体请求进行查询# 向 ES 服务器发 GET请求 http://127.0.0.1:9200/shopping/_search附带JSON体如下 {query:{match:{category:小米}} }# 带请求体方式的查找所有内容 {query:{match_all:{}} }# 查询指定字段 {query:{match_all:{}},_source:[title] }# 分页查询提示from(n-1)*sizesize:一页的数量 {query:{match_all:{}},from:0,size:2 }# 查询排序 {query:{match_all:{}},sort:{price:{order:desc}} } 3.6 文档多条件查询 范围查询 # 向 ES 服务器发 GET请求 http://127.0.0.1:9200/shopping/_search附带JSON体 # bool表示条件的意思must表示都要满足 # 多条件查询 {query:{bool:{must:[{match:{category:小米}},{match:{price:3999.00}}]}} } # 或者的条件查询(should) {query:{bool:{should:[{match:{category:小米}},{match:{category:华为}}]}} }# 加上价格大于5000的条件 {query:{bool:{must:[{match:{category:小米}},{match:{price:3999.00}}],filter:{range:{price:{gt:5000}}}}} } 3.7 文档全文检索 完全匹配 高亮查询 # 向 ES 服务器发 GET请求 http://127.0.0.1:9200/shopping/_search附带JSON体 # 全文检索,这功能像搜索引擎那样如查询品牌输入小华,返回结果品牌带有小、华两个字的数据 {query:{match:{category : 小华}} } # 完全匹配 {query:{match_phrase:{category : 为}} } # 高亮查询 {query:{match_phrase:{category : 为}},highlight:{fields:{category:{}//----高亮category 字段}} } 3.8 文档聚合查询 聚合允许使用者对 es 文档进行统计分析类似与关系型数据库中的 group by当然还有很多其他的聚合例如取最大值max、平均值avg等等 # 按price字段进行分组 {aggs:{//聚合操作price_group:{//名称随意起名terms:{//分组field:price//分组字段}}} } # 上面返回结果会附带原始数据的。若不想要不附带原始数据的结果 {aggs:{price_group:{terms:{//分组field:price}}},size:0//不要附带原始数据 }# 对所有手机价格求平均值 {aggs:{price_avg:{//名称随意起名avg:{//求平均field:price}}},size:0 } 3.9 文档映射关系 在之前的查询有的字段可以分词查询有的不可以我们该如何界定和控制呢创建数据库表需要设置字段名称类型长度约束等索引库也一样需要知道这个类型下有哪些字段每个字段有哪些约束信息这就叫做映射(mapping) 先创建一个索引PUT http://127.0.0.1:9200/user # 创建映射 # 向 ES 服务器发 PUT请求 http://127.0.0.1:9200/user/_mapping附带JSON体 {properties: {name:{type: text,//可以分词index: true //可以索引查询},sex:{type: keyword,//关键字不可以分词index: true},tel:{type: keyword,index: false}} } # 查询映射 http://127.0.0.1:9200/user/_mapping # 增加数据 http://127.0.0.1:9200/user/_create/1001 {name:小米,sex:男的,tel:1111 }# 查找name含有”小“数据,http://127.0.0.1:9200/user/_search {query:{match:{name:小}} } 结果验证查询name为小的时候可以查出来查询sex为男查不出来数据这是因为name type 设置为text可以分词查询sex没有电话index设置为false 4、JavaAPI 4.1 环境准备 新建工程引入依赖 dependenciesdependencygroupIdorg.elasticsearch/groupIdartifactIdelasticsearch/artifactIdversion7.8.0/version/dependency!-- elasticsearch 的客户端 --dependencygroupIdorg.elasticsearch.client/groupIdartifactIdelasticsearch-rest-high-level-client/artifactIdversion7.8.0/version/dependencydependencygroupIdcommons-logging/groupIdartifactIdcommons-logging-api/artifactIdversion1.1/version/dependency!-- elasticsearch 依赖 2.x 的 log4j --dependencygroupIdorg.apache.logging.log4j/groupIdartifactIdlog4j-api/artifactIdversion2.8.2/version/dependencydependencygroupIdorg.apache.logging.log4j/groupIdartifactIdlog4j-core/artifactIdversion2.8.2/version/dependencydependencygroupIdcom.fasterxml.jackson.core/groupIdartifactIdjackson-databind/artifactIdversion2.9.9/version/dependency!-- junit 单元测试 --dependencygroupIdjunit/groupIdartifactIdjunit/artifactIdversion4.12/version/dependencydependencygroupIdorg.projectlombok/groupIdartifactIdlombok/artifactIdversion1.18.24/version/dependency/dependencies ESTest_Client public class ESTest_Client {public static void main(String[] args) throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));// 关闭客户端连接client.close();} } 4.2 索引-创建 public class ESTest {//创建索引Testpublic void ESTest_Client_create() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//创建索引CreateIndexRequest index new CreateIndexRequest(user);CreateIndexResponse createIndexResponse client.indices().create(index, RequestOptions.DEFAULT);//响应状态boolean acknowledged createIndexResponse.isAcknowledged();System.out.println(创建索引acknowledged);// 关闭客户端连接client.close();} }4.3 索引查询 删除 //索引查询 Test public void ESTest_Client_search() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));GetIndexRequest request new GetIndexRequest(user);GetIndexResponse response client.indices().get(request, RequestOptions.DEFAULT);//相应状态System.out.println(response.getAliases());System.out.println(response.getMappings());System.out.println(response.getSettings());// 关闭客户端连接client.close(); } //删除索引 Test public void ESTest_Client_delete() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));DeleteIndexRequest request new DeleteIndexRequest(user);AcknowledgedResponse response client.indices().delete(request, RequestOptions.DEFAULT);//相应状态System.out.println(response.isAcknowledged());// 关闭客户端连接client.close(); } 4.4 文档新增 修改 首先创建数据 Data AllArgsConstructor public class User {private String name;private Integer age;private String sex;} 新增和修改 //插入文档 Test public void ESTest_Client_insert() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//插入数据IndexRequest request new IndexRequest();request.index(user).id(1001);User user new User(张三,30,男);//像ES插入数据必须将数据转化为JSON格式ObjectMapper mapper new ObjectMapper();String userJson mapper.writeValueAsString(user);request.source(userJson, XContentType.JSON);IndexResponse response client.index(request, RequestOptions.DEFAULT);System.out.println(response.getResult());// 关闭客户端连接client.close(); } //修改文档 Test public void ESTest_Client_update() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//插入数据UpdateRequest request new UpdateRequest();request.index(user).id(1001);request.doc(XContentType.JSON,sex,女);UpdateResponse response client.update(request, RequestOptions.DEFAULT);System.out.println(response.getResult());// 关闭客户端连接client.close(); }//查询文档数据 Test public void ESTest_Client_get() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));GetRequest getRequest new GetRequest();getRequest.index(user).id(1001);//插入数据GetResponse response client.get(getRequest, RequestOptions.DEFAULT);System.out.println(response.getSourceAsString());// 关闭客户端连接client.close(); }//删除文档数据 Test public void ESTest_Client_delete_document() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));DeleteRequest deleteRequest new DeleteRequest();deleteRequest.index(user).id(1001);DeleteResponse response client.delete(deleteRequest, RequestOptions.DEFAULT);// 关闭客户端连接client.close(); } 4.5 文档-批量查询 批量删除 //批量新增数据 Test public void ESTest_Client_insert_batch_document() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//批量插入数据BulkRequest bulkRequest new BulkRequest();bulkRequest.add(new IndexRequest().index(user).id(1001).source(XContentType.JSON,name,zhangsan));bulkRequest.add(new IndexRequest().index(user).id(1002).source(XContentType.JSON,name,lisi));bulkRequest.add(new IndexRequest().index(user).id(1003).source(XContentType.JSON,name,wangwu));BulkResponse response client.bulk(bulkRequest, RequestOptions.DEFAULT);System.out.println(response.getTook());System.out.println(response.getItems());// 关闭客户端连接client.close(); }//批量删除数据 Test public void ESTest_Client_delete_batch_document() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//批量删除数据BulkRequest bulkRequest new BulkRequest();bulkRequest.add(new DeleteRequest().index(user).id(1001));bulkRequest.add(new DeleteRequest().index(user).id(1002));bulkRequest.add(new DeleteRequest().index(user).id(1003));BulkResponse response client.bulk(bulkRequest, RequestOptions.DEFAULT);System.out.println(response.getTook());System.out.println(response.getItems());// 关闭客户端连接client.close(); } 4.6 文档-高级查询-全量 //全量查询 Test public void ESTest_Client_query_document() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//1.查询索引中全部的数据SearchRequest request new SearchRequest();request.indices(user);request.source(new SearchSourceBuilder().query(QueryBuilders.matchAllQuery()));SearchResponse response client.search(request, RequestOptions.DEFAULT);SearchHits hits response.getHits();System.out.println(hits.getTotalHits());System.out.println(response.getTook());for (SearchHit hit : hits) {System.out.println(hit.getSourceAsString());}// 关闭客户端连接client.close(); } 4.7 文档-高级查询-termQuery查询、分页查询、排序查询、过滤字段 //条件查询 Test public void ESTest_Client_query_document2() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//1.查询索引中全部的数据SearchRequest request new SearchRequest();request.indices(user);//termQuery 查询//request.source(new SearchSourceBuilder().query(QueryBuilders.termQuery(sex,男)));SearchSourceBuilder builder new SearchSourceBuilder().query(QueryBuilders.matchAllQuery());//分页查询//(当前页码-1)*页数//builder.from(0);//builder.size(2);//排序查询//builder.sort(age, SortOrder.DESC);//过滤字段String[] excludes {age};String[] includes {name,age};builder.fetchSource(includes,excludes);request.source(builder);SearchResponse response client.search(request, RequestOptions.DEFAULT);SearchHits hits response.getHits();System.out.println(hits.getTotalHits());System.out.println(response.getTook());for (SearchHit hit : hits) {System.out.println(hit.getSourceAsString());}// 关闭客户端连接client.close(); } 4.8 文档-高级查询-组合查询 and、or、范围查询 //组合查询 Test public void ESTest_Client_compose_query_document2() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));//1.查询索引中全部的数据SearchRequest request new SearchRequest();request.indices(user);SearchSourceBuilder builder new SearchSourceBuilder();// and or 查询//BoolQueryBuilder boolQueryBuilder QueryBuilders.boolQuery(); // and 的关系//boolQueryBuilder.must(QueryBuilders.matchQuery(age,30));// boolQueryBuilder.must(QueryBuilders.matchQuery(sex,男)); // 或者的关系//boolQueryBuilder.should(QueryBuilders.matchQuery(age,30));//boolQueryBuilder.should(QueryBuilders.matchQuery(age,40));//范围查询RangeQueryBuilder rangeQuery QueryBuilders.rangeQuery(age);rangeQuery.gte(30); // 大于等于rangeQuery.lte(40); // 小于等于builder.query(rangeQuery);request.source(builder);SearchResponse response client.search(request, RequestOptions.DEFAULT);SearchHits hits response.getHits();System.out.println(hits.getTotalHits());System.out.println(response.getTook());for (SearchHit hit : hits) {System.out.println(hit.getSourceAsString());}// 关闭客户端连接client.close(); } 4.9 文档-高级查询-模糊、高亮查询 //模糊查询 Test public void ESTest_Client_like_query_document2() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));SearchRequest request new SearchRequest();request.indices(user);SearchSourceBuilder builder new SearchSourceBuilder();// 模糊查询 Fuzziness.ONE 差一个字符也能查出来// builder.query(QueryBuilders.fuzzyQuery(name,wangwu).fuzziness(Fuzziness.ONE));//request.source(builder);//高亮显示TermQueryBuilder termQueryBuilder QueryBuilders.termQuery(name, zhangsan);HighlightBuilder highlightBuilder new HighlightBuilder();highlightBuilder.preTags(font colorred);highlightBuilder.postTags(/font);highlightBuilder.field(name);builder.highlighter(highlightBuilder);builder.query(termQueryBuilder);request.source(builder);SearchResponse response client.search(request, RequestOptions.DEFAULT);System.out.println(response.toString());SearchHits hits response.getHits();System.out.println(hits.getTotalHits());System.out.println(response.getTook());for (SearchHit hit : hits) {System.out.println(hit.getSourceAsString());}// 关闭客户端连接client.close(); } 4.10 文档-聚合查询 //聚合查询 Test public void ESTest_Client_polymerization_query_document() throws IOException {//创建客户端RestHighLevelClient client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));SearchRequest request new SearchRequest();request.indices(user);SearchSourceBuilder builder new SearchSourceBuilder();//最大值查询//AggregationBuilder aggregationBuilder AggregationBuilders.max(maxAge).field(age);//builder.aggregation(aggregationBuilder);//分组查询AggregationBuilder aggregationBuilder AggregationBuilders.terms(ageGroup).field(age);builder.aggregation(aggregationBuilder);request.source(builder);SearchResponse response client.search(request, RequestOptions.DEFAULT);System.out.println(response.toString());SearchHits hits response.getHits();System.out.println(hits.getTotalHits());System.out.println(response.getTook());for (SearchHit hit : hits) {System.out.println(hit.getSourceAsString());}// 关闭客户端连接client.close(); } 三、ElasticSearch 环境 1、相关概念 1.1 单机 集群 单台ElasticSearch 服务器提供服务往往都有最大的负载能力超过这个阈值服务器性能就会大大降低甚至不可用所以生产环境中一般都是运行在指定服务器集群中。除了负载能力单点服务器也存在其他问题 单台机器存储容量有限单服务器容易出现单点故障无法实现高可用单服务的并发能力有限 配置服务器集群时集群中节点数量没有限制大于等于 2 个节点就可以看做是集群了。一 般出于高性能及高可用方面来考虑集群中节点数量都是 3 个以上 1.2 集群Cluster 一个集群就是由一个或多个服务器节点组织在一起共同持有整个的数据并一起提供索引和搜索功能。一个 Elasticsearch 集群有一个唯一的名字标识这个名字默认就是elasticsearch。这个名字是重要的因为一个节点只能通过指定某个集群的名字来加入这个集群 1.3 节点 Node 集群中包含很多服务器 一个节点就是其中的一个服务器。 作为集群的一部分它存储数据参与集群的索引和搜索功能。 一个节点也是由一个名字来标识的默认情况下这个名字是一个随机的漫威漫画角色的名字这个名字会在启动的时候赋予节点。这个名字对于管理工作来说挺重要的因为在这个管理过程中你会去确定网络中的哪些服务器对应于 Elasticsearch 集群中的哪些节点。一个节点可以通过配置集群名称的方式来加入一个指定的集群。默认情况下每个节点都会被安排加入到一个叫做**“elasticsearch”**的集群中这意味着如果你在你的网络中启动了若干个节点并假定它们能够相互发现彼此它们将会自动地形成并加入到一个叫做elasticsearch的集群中。 在一个集群里只要你想可以拥有任意多个节点。而且如果当前你的网络中没有运行任何 Elasticsearch 节点这时启动一个节点会默认创建并加入一个叫做elasticsearch的集群。 2、Windows集群 创建elasticsearch-cluster文件夹在内部复制三个elasticsearch服务(要全新的解压分别命名node-1001/1002/1003) 修改集群文件目录中每个节点的config/elasticsearch.yml 配置文件 node-1001 节点 # Elasticsearch Configuration # # NOTE: Elasticsearch comes with reasonable defaults for most settings. # Before you set out to tweak and tune the configuration, make sure you # understand what are you trying to accomplish and the consequences. # # The primary way of configuring a node is via this file. This template lists # the most important settings you may want to configure for a production cluster. # # Please consult the documentation for further information on configuration options: # https://www.elastic.co/guide/en/elasticsearch/reference/index.html # # ---------------------------------- Cluster ----------------------------------- # # Use a descriptive name for your cluster: #集群名称、节点之间要保持一致 cluster.name: my-application # # ------------------------------------ Node ------------------------------------ # # Use a descriptive name for the node: # #节点名称集群内要唯一 node.name: node-1001 node.master: true node.data: true # # Add custom attributes to the node: # #node.attr.rack: r1 # # ----------------------------------- Paths ------------------------------------ # # Path to directory where to store the data (separate multiple locations by comma): # #path.data: /path/to/data # # Path to log files: # #path.logs: /path/to/logs # # ----------------------------------- Memory ----------------------------------- # # Lock the memory on startup: # #bootstrap.memory_lock: true # # Make sure that the heap size is set to about half the memory available # on the system and that the owner of the process is allowed to use this # limit. # # Elasticsearch performs poorly when the system is swapping the memory. # # ---------------------------------- Network ----------------------------------- # # Set the bind address to a specific IP (IPv4 or IPv6): #ip地址 network.host: localhost # # Set a custom port for HTTP: #http端口 http.port: 1001 #tcp 监听端口 transport.tcp.port: 9301 # # For more information, consult the network module documentation. # # --------------------------------- Discovery ---------------------------------- # # Pass an initial list of hosts to perform discovery when this node is started: # The default list of hosts is [127.0.0.1, [::1]] # #discovery.seed_hosts: [host1, host2] # # Bootstrap the cluster using an initial set of master-eligible nodes: # #cluster.initial_master_nodes: [node-1, node-2] # # For more information, consult the discovery and cluster formation module documentation. # # ---------------------------------- Gateway ----------------------------------- # # Block initial recovery after a full cluster restart until N nodes are started: # #gateway.recover_after_nodes: 3 # # For more information, consult the gateway module documentation. # # ---------------------------------- Various ----------------------------------- # # Require explicit names when deleting indices: # #action.destructive_requires_name: true # 跨域配置 http.cors.enabled: true http.cors.allow-origin: * node-1002 节点自行增删 #节点名称集群内要唯一 node.name: node-1002 node.master: true node.data: true#ip地址 network.host: localhost # # Set a custom port for HTTP: #http端口 http.port: 1002 #tcp 监听端口 transport.tcp.port: 9302# 这是发现第一台的配置 discovery.seed_hosts: [localhost:9301] discovery.zen.fd.ping_timeout: 1m discovery.zen.fd.ping_retries: 5# 跨域配置 http.cors.enabled: true http.cors.allow-origin: * node-1003 节点 cluster.name: my-application#节点名称集群内要唯一 node.name: node-1003 node.master: true node.data: true#ip地址 network.host: localhost # # Set a custom port for HTTP: #http端口 http.port: 1003 #tcp 监听端口 transport.tcp.port: 9303discovery.seed_hosts: [localhost:9301,localhost:9302] discovery.zen.fd.ping_timeout: 1m discovery.zen.fd.ping_retries: 5#action.destructive_requires_name: true # 跨域配置 http.cors.enabled: true http.cors.allow-origin: * 最后启动前先删除每个节点中的data目录中所有内容如果存在然后分别双击执行bin/elsaticsearch.bat启动节点服务器会自动加入指定名称的集群查询集群状态localhost:1001/_cluster/health 3、Linux单机 Linux下载地址https://www.elastic.co/cn/downloads/past-releases/elasticsearch-7-8-0 # 将下载的软件解压缩 #解压缩 tar -axvf elasticsearch-7.8.0-linux-x86_64.tar.gz # 改名称 mv elasticsearch-7.8.0 es# 创建用户,因为安全问题Elasticsearch不允许root用户直接运行所以创建新用户在root用户中创建新用户 #新增es用户 useradd es #为es用户设置密码 passwd es # 如果错误了可以删除再加 userdel -r es #文件夹所有者 chown -R es:es /usr/local/elasticsearch/single/es# 修改配置文件 # 修改/usr/local/elasticsearch/single/es/config/elasticsearch.yml #加入如下配置 cluster.name: elasticsearch node.name: node-1 network.host: 0.0.0.0 cluster.initial_master_nodes: [node-1]# 由于es文件比较多需要修改系统配置 # 修改 /etc/security/limits.conf # 在文件末尾中增加下面内容每个进程可以打开的文件数的限制 es soft nofile 65536 es hard nofile 65536 # 修改 /etc/security/limits.d/20-nproc.conf # 在文件末尾中增加下面内容每个进程可以打开的文件数的限制 es soft nofile 65536 es hard nofile 65536 # 操作系统级别对每个用户创建的进程数的限制 * hard nproc 4096 # 注 * 地表linux 所有用户名称 # 修改/etc/sysctl.conf # 在文件末尾中增加下面内容 # 一个进程可以拥有的 VMA(虚拟内存区域)的数量,默认值为 65536 vm.max_map_count655360# 重新加载 sysctl -p# 最后启动软件 #切换es用户 su es #启动ES bin/elasticsearch #如果出现 Likely root cause: java.nio.file.AccessDeniedException: /usr/local/elasticsearch/single/es/config/elasticsearch.keystore错误 # 切换root用户 chown -R es:es /usr/local/elasticsearch/single/es 4、Linux集群 #解压缩 tar -axvf elasticsearch-7.8.0-linux-x86_64.tar.gz # 改名称 mv elasticsearch-7.8.0 es-cluster # 最后将软件分发到其他节点(可以使用xsync)# 因为安全问题Elasticsearch不允许root用户直接运行所以创建新用户在root用户中创建新用户 #新增es用户 useradd es #为es用户设置密码 passwd es # 如果错误了可以删除再加 userdel -r es #文件夹所有者 chown -R es:es /usr/local/elasticsearch/single/es 然后修改/usr/local/elasticsearch/cluster/es-cluster/config/elasticsearch.yml 文件 # 加入如下配置 #集群名称 cluster.name: cluster-es #节点名称 每个节点的名称不能重复 node.name: node-1 #ip 地址 每个节点的地址不能重复 network.host: node01 #是不是有资格主节点 node.master: true node.data: true http.port: 9200 # head 插件需要这打开这两个配置 http.cors.allow-origin: * http.cors.enabled: true http.max_content_length: 200mb #es7.x 之后新增的配置初始化一个新的集群时需要此配置来选举 master cluster.initial_master_nodes: [node-1] #es7.x 之后新增的配置节点发现 discovery.seed_hosts: [node01:9300,node02:9300,node03:9300] gateway.recover_after_nodes: 2 network.tcp.keep_alive: true network.tcp.no_delay: true transport.tcp.compress: true #集群内同时启动的数据任务个数默认是 2 个 cluster.routing.allocation.cluster_concurrent_rebalance: 16 #添加或删除节点及负载均衡时并发恢复的线程个数默认 4 个 cluster.routing.allocation.node_concurrent_recoveries: 16 #初始化数据恢复时并发恢复线程的个数默认 4 个 cluster.routing.allocation.node_initial_primaries_recoveries: 16 然后系统的配置如同单机最后启动软件分别在不同节点上启动ES软件 cd /usr/local/elasticsearch/cluster/es-cluster #启动 bin/slasticsearch #后台启动 bin/slasticsearch -d# 测试集群 http://node01:9200/_cat/nodes 四、Elasticsearch进阶 1、核心概念 1.1 索引index 一个索引就是一个拥有几分相似特征的文档的集合。比如说你可以有一个客户数据的索引另一个产品目录的索引还有一个订单数据的索引。一个索引由一个名字来标识必须全部是小写字母并且当我们要对这个索引中的文档进行索引、搜索、更新和删除CRUD的时候都要使用到这个名字。在一个集群中可以定义任意多的索引。 能搜索的数据必须索引这样的好处是可以提高查询速度比如新华字典前面的目录就是索引的意思目录可以提高查询速度。 Elasticsearch 索引的精髓一切设计都是为了提高搜索的性能。 1.2 类型 type 在一个索引中你可以定义一种或多种类型。一个类型是你的索引的一个逻辑上的分类/分区其语义完全由你来定。通常会为具有一组共同字段的文档定义一个类型。不同的版本类型发生了不同的变化。 版本Type5.x支持多种type6.x只能有一种type7.x默认不再支持自定义索引类型默认类型为 _doc 1.3 文档Document 一个文档是一个可被索引的基础信息单元也就是一条数据 比如你可以拥有某一个客户的文档某一个产品的一个文档当然也可以拥有某个订单的一个文档。文档以 JSONJavascript Object Notation格式来表示而 JSON 是一个到处存在的互联网数据交互格式。在一个 index/type 里面你可以存储任意多的文档。 1.4 字段Field 相当于是数据表的字段对文档数据根据不同属性进行的分类标识 1.5 映射Mapping mapping 是处理数据的方式和规则方面做一些限制如某个字段的数据类型、默认值、分析器、是否被索引等等。这些都是映射里面可以设置的其它就是处理 ES 里面数据的一些使用规则设置也叫做映射按着最优规则处理数据对性能提高很大因此才需要建立映射并且需要思考如何建立映射才能对性能更好。 1.6 分片Shards 一个索引可以存储超出单个节点硬件限制的大量数据。比如一个具有 10 亿文档数据 的索引占据 1TB 的磁盘空间而任一节点都可能没有这样大的磁盘空间。 或者单个节点处理搜索请求响应太慢。为了解决这个问题Elasticsearch 提供了将索引划分成多份的能力每一份就称之为分片。当你创建一个索引的时候你可以指定你想要的分片的数量。每个分片本身也是一个功能完善并且独立的“索引”这个“索引”可以被放置到集群中的任何节点上。分片很重要主要有两方面的原因 允许你水平分割 / 扩展你的内容容量允许你在分片之上进行分布式的、并行的操作进而提高性能/吞吐量 至于一个分片怎样分布它的文档怎样聚合和搜索请求是完全由 Elasticsearch 管理的对于作为用户的你来说这些都是透明的无需过分关心。被混淆的概念是一个 Lucene 索引 我们在 Elasticsearch 称作 分片 。 一个Elasticsearch 索引 是分片的集合。 当 Elasticsearch 在索引中搜索的时候 他发送查询到每一个属于索引的分片Lucene 索引然后合并每个分片的结果到一个全局的结果集。 1.7 副本Replicas 在一个网络 / 云的环境里失败随时都可能发生在某个分片/节点不知怎么的就处于离线状态或者由于任何原因消失了这种情况下有一个故障转移机制是非常有用并且是强烈推荐的。为此目的 Elasticsearch 允许你创建分片的一份或多份拷贝这些拷贝叫做复制分片(副本)。 在分片/节点失败的情况下提供了高可用性。因为这个原因注意到复制分片从不与原/主要original/primary分片置于同一节点上是非常重要的。扩展你的搜索量/吞吐量因为搜索可以在所有的副本上并行运行。 总之每个索引可以被分成多个分片。一个索引也可以被复制 0 次意思是没有复制或多次。一旦复制了每个索引就有了主分片作为复制源的原来的分片和复制分片主分片的拷贝之别。 分片和复制的数量可以在索引创建的时候指定。在索引创建之后你可以在任何时候动态地改变复制的数量但是你事后不能改变分片的数量。默认情况下Elasticsearch 中的每个索引被分片 1 个主分片和 1 个复制这意味着如果你的集群中至少有两个节点你的索引将会有 1 个主分片和另外 1 个复制分片1 个完全拷贝这样的话每个索引总共就有 2 个分片 我们需要根据索引需要确定分片个数。 1.8 分配Allocation 将分片分配给某个节点的过程包括分配主分片或者副本。如果是副本还包含从主分片复制数据的过程。这个过程是由 master 节点完成的。 2、系统架构 一个运行中的 Elasticsearch 实例称为一个节点而集群是由一个或者多个拥有相同 cluster.name 配置的节点组成 它们共同承担数据和负载的压力。当有节点加入集群中或者从集群中移除节点时集群将会重新平均分布所有的数据。 当一个节点被选举成为主节点时 它将负责管理集群范围内的所有变更例如增加、 删除索引或者增加、删除节点等。 而主节点并不需要涉及到文档级别的变更和搜索等操作所以当集群只拥有一个主节点的情况下即使流量的增加它也不会成为瓶颈。 任何节点都可以成为主节点。我们的示例集群就只有一个节点所以它同时也成为了主节点。 作为用户我们可以将请求发送到集群中的任何节点 包括主节点。 每个节点都知道 任意文档所处的位置并且能够将我们的请求直接转发到存储我们所需文档的节点。 无论我们将请求发送到哪个节点它都能负责从各个包含我们所需文档的节点收集回数据并将最终结果返回給客户端。 Elasticsearch 对这一切的管理都是透明的。 3、分布式集群 3.1 单节点集群 我们在包含一个空节点的集群内创建名为user的索引为了掩饰目的我们将分配3个主分片和一个副本每个主分片拥有一个副本分片集群现在是拥有一个索引的单节点集群。所有 3 个主分片都被分配在 node-1 #PUT http://127.0.0.1:1001/users {settings : {number_of_shards : 3,number_of_replicas : 1} } 通过 elasticsearch-head 插件一个Chrome插件查看集群情况 集群健康值:yellow( 3 of 6 )表示当前集群的全部主分片都正常运行但是副本分片没有全部处在正常状态(当前集群是正常运行的但存在丢失数据的风险。)3 个主分片正常。3 个副本分片都是 Unassigned它们都没有被分配到任何节点。 在同 一个节点上既保存原始数据又保存副本是没有意义的因为一旦失去了那个节点我们也将丢失该节点 上的所有副本数据 3.2 故障转移 当集群中只有一个节点在运行时意味着会有一个单点故障问题——没有冗余。 幸运的是我们只需再启动一个节点即可防止数据丢失。当你在同一台机器上启动了第二个节点时只要它和第一个节点有同样的 cluster.name 配置它就会自动发现集群并加入到其中。但是在不同机器上启动节点的时候为了加入到同一集群你需要配置一个可连接到的单播主机列表。之所以配置为使用单播发现以防止节点无意中加入集群。只有在同一台机器上运行的节点才会自动组成集群。 如果启动了第二个节点集群将会拥有两个节点 : 所有主分片和副本分片都已被分配 3.3 水平扩容 怎样为我们的正在增长中的应用程序按需扩容呢当启动了第三个节点我们的集群将会拥有三个节点的集群 : **为了分散负载而对分片进行重新分配 **。 Node 1 和 Node 2 上各有一个分片被迁移到了新的 Node 3 节点现在每个节点上都拥有 2 个分片 而不是之前的 3 个。 这表示每个节点的硬件资源CPU, RAM, I/O将被更少的分片所共享每个分片 的性能将会得到提升 分片是一个功能完整的搜索引擎它拥有使用一个节点上的所有资源的能力。 我们这个拥有 6 个分 片3 个主分片和 3 个副本分片的索引可以最大扩容到 6 个节点每个节点上存在一个分片并且每个分片拥有所在节点的全部资源。 但是如果我们想要扩容超过 6 个节点怎么办呢 主分片的数目在索引创建时就已经确定了下来。实际上这个数目定义了这个索引能够存储 的最大数据量。实际大小取决于你的数据、硬件和使用场景。 但是读操作——搜索和返回数据——可以同时被主分片 或 副本分片所处理所以当你拥有越多的副本分片时也将拥有越高的吞吐量。在运行中的集群上是可以动态调整副本分片数目的我们可以按需伸缩集群。让我们把副本数从默认的 1 增加到 2。 #PUT http://127.0.0.1:1001/users/_settings {number_of_replicas : 2 } users 索引现在拥有 9 个分片 3 个主分片和 6 个副本分片。 这意味着我们可以将集群扩容到 9 个节点每个节点上一个分片。相比原来 3 个节点时集群搜索性能可以提升 3 倍。 3.4 应对故障 虽然我们拥有所有的三个主分片但是同时设置了每个主分片需要对应 2 份副本分片而此时只存在一份副本分片。 所以集群不能为 green 的状态不过我们不必过于担心如果我们同样关闭了 Node 2 我们的程序 依然 可以保持在不丢任何数据的情况下运行因为Node 3 为每一个分片都保留着一份副本。 如果想回复原来的样子要确保Node-1的配置文件有如下配置让他找到集群 discovery.seed_hosts: [localhost:9302, localhost:9303] 集群可以将缺失的副本分片再次进行分配那么集群的状态也将恢复成之前的状态。 如果 Node 1 依然拥有着之前的分片它将尝试去重用它们同时仅从主分片复制发生了修改的数据文件。和之前的集群相比只是 Master 节点切换了 3.5 路由计算 分片控制 路由计算 当索引一个文档的时候文档会被存储到一个主分片中。 Elasticsearch 如何知道一个文档应该存放到哪个分片中呢当我们创建文档时它如何决定这个文档应当被存储在分片 1 还是分片 2 中呢首先这肯定不会是随机的否则将来要获取文档的时候我们就不知道从何处寻找了。实际上这个过程是根据下面这个公式决定的shard hash(routing) % number_of_primary_shards 所有的文档API ( get ,index ,delete ,bulk , update以及 mget 都接受一个叫做routing 的路由参数通过这个参数我们可以自定义文档到分片的映射。一个自定义的路由参数可以用来确保所有相关的文档—例如所有属于同一个用户的文档——都被存储到同一个分片中 分片控制 我们可以发送请求到集群中的任一节点。每个节点都有能力处理任意请求。每个节点都知道集群中任一文档位置所以可以直接将请求转发到需要的节点上。在下面的例子中如果将所有的请求发送到Node 1001我们将其称为协调节点coordinating node 当发送请求的时候 为了扩展负载更好的做法是轮询集群中所有的节点。 3.6 数据写流程 新建、索引和删除请求都是写操作 必须在主分片上面完成之后才能被复制到相关的副本分片 在客户端收到成功响应时文档变更已经在主分片和所有副本分片执行完成变更是安全的。有一些可选的请求参数允许您影响这个过程可能以数据安全为代价提升性能。这些选项很少使用因为 Elasticsearch 已经很快但是为了完整起见 请参考下文 consistency 即一致性。在默认设置下即使仅仅是在试图执行一个写操作之前主分片都会要求必须要有规定数量quorum或者换种说法也即必须要有大多数的分片副本处于活跃可用状态才会去执行写操作其中分片副本 可以是主分片或者副本分片。这是为了避免在发生网络分区故障network partition的时候进行写操作进而导致数据不一致。 规定数量即 int((primary number_of_replicas) / 2 ) 1consistency 参数的值可以设为 one 只要主分片状态 ok 就允许执行写操作all必须要主分片和所有副本分片的状态没问题才允许执行写操作quorum默认值为quorum , 即大多数的分片副本状态没问题就允许执行写操作 注意规定数量的计算公式中number_of_replicas指的是在索引设置中的设定副本分片数而不是指当前处理活动状态的副本分片数。如果你的索引设置中指定了当前索引拥有3个副本分片那规定数量的计算结果即int((1 primary 3 replicas) / 2) 1 3如果此时你只启动两个节点那么处于活跃状态的分片副本数量就达不到规定数量也因此您将无法索引和删除任何文档 timeout 如果没有足够的副本分片会发生什么Elasticsearch 会等待希望更多的分片出现。默认情况下它最多等待 1 分钟。 如果你需要你可以使用timeout参数使它更早终止100是100 毫秒30s是30秒。 新索引默认有1个副本分片这意味着为满足规定数量应该需要两个活动的分片副本。 但是这些默认的设置会阻止我们在单一节点上做任何事情。为了避免这个问题要求只有当number_of_replicas 大于1的时候规定数量才会执行 3.7 数据读流程 在处理读取请求时协调结点在每次请求的时候都会通过轮询所有的副本分片来达到负载均衡。在文档被检索时已经被索引的文档可能已经存在于主分片上但是还没有复制到副本分片。 在这种情况下副本分片可能会报告文档不存在但是主分片可能成功返回文档。 一旦索引请求成功返回给用户文档在主分片和副本分片都是可用的。 3.8 更新流程 批量操作流程 更新流程部分更新一个文档结合了先前说明的读取和写入流程 客户端向Node 1发送更新请求它将请求转发到主分片所在的Node 3 Node 3从主分片检索文档修改_source字段中的JSON并且尝试重新索引主分片的文档。如果文档已经被另一个进程修改,它会重试步骤3 ,超过retry_on_conflict次后放弃如果 Node 3成功地更新文档它将新版本的文档并行转发到Node 1和 Node 2上的副本分片重新建立索引。一旦所有副本分片都返回成功Node 3向协调节点也返回成功协调节点向客户端返回成功 批量操作流程 **mget和 bulk API的模式类似于单文档模式。**区别在于协调节点知道每个文档存在于哪个分片中。它将整个多文档请求分解成每个分片的多文档请求并且将这些请求并行转发到每个参与节点。协调节点一旦收到来自每个节点的应答就将每个节点的响应收集整理成单个响应返回给客户端。 用单个 mget 请求取回多个文档所需的步骤顺序: 客户端向 Node 1 发送 mget 请求。Node 1为每个分片构建多文档获取请求然后并行转发这些请求到托管在每个所需的主分片或者副本分片的节点上。一旦收到所有答复Node 1 构建响应并将其返回给客户端。 可以对docs数组中每个文档设置routing参数。bulk API 允许在单个批量请求中执行多个创建、索引、删除和更新请求 bulk API 按如下步骤顺序执行 客户端向Node 1 发送 bulk请求Node 1为每个节点创建一个批量请求并将这些请求并行转发到每个包含主分片的节点主机主分片一个接一个按顺序执行每个操作。当每个操作成功时,主分片并行转发新文档或删除到副本分片然后执行下一个操作。一旦所有的副本分片报告所有操作成功该节点将向协调节点报告成功协调节点将这些响应收集整理并返回给客户端。 4、分片原理 分片是Elasticsearch最小的工作单元。但是究竟什么是一个分片它是如何工作的传统的数据库每个字段存储单个值但这对全文检索并不够。文本字段中的每个单词需要被搜索对数据库意味着需要单个字段有索引多值的能力。最好的支持是一个字段多个值需求的数据结构是倒排索引。 4.1 倒排索引 Elasticsearch使用一种称为倒排索引的结构它适用于快速的全文搜索。见其名知其意有倒排索引肯定会对应有正向索引。正向索引forward index反向索引inverted index更熟悉的名字是倒排索引。倒排索引Inverted Index也叫反向索引有反向索引必有正向索引。通俗地来讲正向索引是通过key找value反向索引则是通过value找key Term单词一段文本经过分析器分析以后就会输出一串单词这一个一个的就叫做Term直译为单词Term Dictionary单词字典顾名思义它里面维护的是Term可以理解为Term的集合Term Index单词索引为了更快的找到某个单词我们为单词建立索引Posting List倒排列表倒排列表记录了出现过某个单词的所有文档的文档列表及单词在该文档中出现的位置信息每条记录称为一个倒排项(Posting)。根据倒排列表即可获知哪些文档包含某个单词。PS实际的倒排列表中并不只是存了文档ID这么简单还有一些其它的信息比如词频Term出现的次数、偏移量offset等可以想象成是Python中的元组或者Java中的对象 索引是为了提高数据的查询速度相当于给数据进行编号在查找数据的时候就可以通过编号快速找到相应的数据。在倒排索引中通过Term索引可以找到Term在Term Dictionary中的位置进而找到Posting List有了倒排列表就可以根据ID找到文档了 4.2 文档搜索 早期的全文检索会为整个文档集合建立一个很大的倒排索引并将其写入到磁盘。 一旦新的索引就绪旧的就会被其替换这样最近的变化便可以被检索到。 倒排索引被写入磁盘后是不可改变的它永远不会修改。不需要锁。如果你从来不更新索引你就不需要担心多进程同时修改数据的问题。一旦索引被读入内核的文件系统缓存便会留在哪里由于其不变性。只要文件系统缓存中还有足够的空间那么大部分读请求会直接请求内存而不会命中磁盘。这提供了很大的性能提升。其它缓存(像filter缓存)在索引的生命周期内始终有效。它们不需要在每次数据改变时被重建因为数据不会变化。写入单个大的倒排索引允许数据被压缩减少磁盘IO和需要被缓存到内存的索引的使用量。 当然一个不变的索引也有不好的地方。主要事实是它是不可变的! 你不能修改它。如果你需要让一个新的文档可被搜索你需要重建整个索引。这要么对一个索引所能包含的数据量造成了很大的限制要么对索引可被更新的频率造成了很大的限制。 4.3 动态更新索引 如何在保留不变性的前提下实现倒排索引的更新答案是用更多的索引。通过增加新的补充索引来反映新近的修改而不是直接重写整个倒排索引。每一个倒排索引都会被轮流查询到,从最早的开始查询完后再对结果进行合并。 Elasticsearch基于Lucene这个java库引入了按段搜索的概念。每一段本身都是一个倒排索引但索引在 Lucene 中除表示所有段的集合外还增加了提交点的概念—一个列出了所有已知段的文件。按段搜索会以如下流程执行 新文档被收集到内存索引缓存不时地, 缓存被提交 一个新的段一个追加的倒排索引被写入磁盘一个新的包含新段名字的提交点被写入磁盘磁盘进行同步所有在文件系统缓存中等待的写入都刷新到磁盘以确保它们被写入物理文件 新的段被开启让它包含的文档可见以被搜索内存缓存被清空等待接收新的文档 当一个查询被触发所有已知的段按顺序被查询。词项统计会对所有段的结果进行聚合以保证每个词和每个文档的关联都被准确计算。这种方式可以用相对较低的成本将新文档添加到索引。段是不可改变的所以既不能从把文档从旧的段中移除也不能修改旧的段来进行反映文档的更新。取而代之的是每个提交点会包含一个.del 文件文件中会列出这些被删除文档的段信息。 当一个**文档被“删除”**时它实际上只是在 .del 文件中被标记删除。一个被标记删除的文档仍然可以被查询匹配到但它会在最终结果被返回前从结果集中移除。文档更新也是类似的操作方式:当一个文档被更新时旧版本文档被标记删除文档的新版本被索引到一个新的段中。可能两个版本的文档都会被一个查询匹配到但被删除的那个旧版本文档在结果集返回前就已经被移除。 4.4 文档刷新 文档刷写 文档合并 近实时搜索 随着按段per-segment搜索的发展一个新的文档从索引到可被搜索的延迟显著降低了。新文档在几分钟之内即可被检索但这样还是不够快。磁盘在这里成为了瓶颈。提交Commiting一个新的段到磁盘需要一个fsync来确保段被物理性地写入磁盘这样在断电的时候就不会丢失数据。但是fsync操作代价很大如果每次索引一个文档都去执行一次的话会造成很大的性能问题。 我们需要的是一个更轻量的方式来使一个文档可被搜索这意味着fsync要从整个过程中被移除。在Elasticsearch和磁盘之间是文件系统缓存。像之前描述的一样在内存索引缓冲区中的文档会被写入到一个新的段中。但是这里新段会被先写入到文件系统缓存—这一步代价会比较低稍后再被刷新到磁盘—这一步代价比较高。不过只要文件已经在缓存中就可以像其它文件一样被打开和读取了 Lucene允许新段被写入和打开使其包含的文档在未进行一次完整提交时便对搜索可见。这种方式比进行一次提交代价要小得多并且在不影响性能的前提下可以被频繁地执行。在 Elasticsearch 中写入和打开一个新段的轻量的过程叫做refresh。默认情况下每个分片会每秒自动刷新一次。这就是为什么我们说 Elasticsearch是近实时搜索文档的变化并不是立即对搜索可见但会在一秒之内变为可见。 这些行为可能会对新用户造成困惑他们索引了一个文档然后尝试搜索它但却没有搜到。这个问题的解决办法是用refresh API执行一次手动刷新/usersl_refresh。尽管刷新是比提交轻量很多的操作它还是会有性能开销。当写测试的时候手动刷新很有用但是不要在生产环境下每次索引一个文档都去手动刷新。相反你的应用需要意识到Elasticsearch 的近实时的性质并接受它的不足。并不是所有的情况都需要每秒刷新。可能你正在使用Elasticsearch索引大量的日志文件你可能想优化索引速度而不是近实时搜索可以通过设置refresh_interval 降低每个索引的刷新频率 {settings: {refresh_interval: 30s} } # refresh_interval可以在既存索引上进行动态更新。 # 在生产环境中当你正在建立一个大的新索引时可以先关闭自动刷新待开始使用该索引时再把它们调回来。 # 关闭自动刷新 PUT /users/_settings { refresh_interval: -1 }# 每一秒刷新 PUT /users/_settings { refresh_interval: 1s } 持久化变更 如果没有用fsync把数据从文件系统缓存刷flush到硬盘我们不能保证数据在断电甚至是程序正常退出之后依然存在。为了保证Elasticsearch 的可靠性需要确保数据变化被持久化到磁盘。在动态更新索引我们说一次完整的提交会将段刷到磁盘并写入一个包含所有段列表的提交点。Elasticsearch 在启动或重新打开一个索引的过程中使用这个提交点来判断哪些段隶属于当前分片。 即使通过每秒刷新(refresh实现了近实时搜索我们仍然需要经常进行完整提交来确保能从失败中恢复。但在两次提交之间发生变化的文档怎么办?我们也不希望丢失掉这些数据。Elasticsearch 增加了一个translog 或者叫事务日志在每一次对Elasticsearch进行操作时均进行了日志记录。整个流程如下: 一个文档被索引之后就会被添加到内存缓冲区并且追加到了 translog 刷新refresh使分片每秒被刷新refresh一次 这些在内存缓冲区的文档被写入到一个新的段中且没有进行fsync操作这个段被打开使其可被搜索内存缓冲区被清空 这个进程继续工作更多的文档被添加到内存缓冲区和追加到事务日志 每隔一段时间—例如translog变得越来越大索引被刷新flush一个新的translog被创建并且一个全量提交被执行 所有在内存缓冲区的文档都被写入一个新的段。缓冲区被清空。一个提交点被写入硬盘。文件系统缓存通过fsync被刷新flush 。老的translog被删除 translog 提供所有还没有被刷到磁盘的操作的一个持久化纪录。当Elasticsearch启动的时候它会从磁盘中使用最后一个提交点去恢复己知的段并且会重放translog 中所有在最后一次提交后发生的变更操作。translog 也被用来提供实时CRUD。当你试着通过ID查询、更新、删除一个文档它会在尝试从相应的段中检索之前首先检查 translog任何最近的变更。这意味着它总是能够实时地获取到文档的最新版本。 段合并 由于自动刷新流程每秒会创建一个新的段这样会导致短时间内的段数量暴增。而段数目太多会带来较大的麻烦。每一个段都会消耗文件句柄、内存和 cpu运行周期。更重要的是每个搜索请求都必须轮流检查每个段所以段越多搜索也就越慢。 Elasticsearch通过在后台进行段合并来解决这个问题。小的段被合并到大的段然后这些大的段再被合并到更大的段。段合并的时候会将那些旧的已删除文档从文件系统中清除。被删除的文档或被更新文档的旧版本不会被拷贝到新的大段中。启动段合并不需要你做任何事。进行索引和搜索时会自动进行。 当索引的时候刷新refresh操作会创建新的段并将段打开以供搜索使用 合并进程选择一小部分大小相似的段并且在后台将它们合并到更大的段中。这并不会中断索引和搜索 一旦合并结束老的段被删除 新的段被刷新(flush)到了磁盘写入一个包含新段且排除旧的和较小的段的新提交点新的段被打开用来搜索。老的段被删除 合并大的段需要消耗大量的 I/O 和 CPU 资源如果任其发展会影响搜索性能。 Elasticsearch在默认情况下会对合并流程进行资源限制所以搜索仍然有足够的资源很好地执行 4.5 文档分析 分析包含下面的过程 将一块文本分成适合于倒排索引的独立的词条将这些词条统一化为标准格式以提高它们的“可搜索性”或者recall 分析器执行上面的工作。分析器实际上是将三个功能封装到了一个包里 字符过滤器首先字符串按顺序通过每个 字符过滤器 。他们的任务是在分词前整理字符串。一个字符过滤器可以用来去掉 HTML或者将 转化成 and。分词器其次字符串被分词器分为单个的词条。一个简单的分词器遇到空格和标点的时候可能会将文本拆分成词条。Token 过滤器最后词条按顺序通过每个 token 过滤器 。这个过程可能会改变词条例如小写化Quick 删除词条例如 像 a and the 等无用词或者增加词条例如像jump和leap这种同义词 内置分析器 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 ,它们对相关性没有多少影响它们会被删除。由于理解英语语法的规则这个分词器可以提取英语单词的词干。英语分词器会产生下面的词条注意看transparent、calling和 set_trans已经变为词根格式 set, shape, semi, transpar, call, set_tran, 5 分析器使用场景 当我们索引一个文档它的全文域被分析成词条以用来创建倒排索引。但是当我们在全文域搜索的时候我们需要将查询字符串通过相同的分析过程以保证我们搜索的词条格式与索引中的词条格式一致。全文查询理解每个域是如何定义的因此它们可以做正确的事 当你查询一个全文域时会对查询字符串应用相同的分析器以产生正确的搜索词条列表当你查询一个精确值域时不会分析查询字符串而是搜索你指定的精确值 测试分析器 有些时候很难理解分词的过程和实际被存储到索引中的词条特别是你刚接触Elasticsearch。为了理解发生了什么你可以使用analyze API来看文本是如何被分析的。在消息体里指定分析器和要分析的文本 #GET http://localhost:9200/_analyze {analyzer: standard,text: Text to analyze } token是实际存储到索引中的词条。start_ offset 和end_ offset指明字符在原始字符串中的位置。position指明词条在原始文本中出现的位置。 指定分析器 当Elasticsearch在你的文档中检测到一个新的字符串域它会自动设置其为一个全文字符串域使用 标准 分析器对它进行分析。你不希望总是这样。可能你想使用一个不同的分析器适用于你的数据使用的语言。有时候你想要一个字符串域就是一个字符串域不使用分析直接索引你传入的精确值例如用户 ID 或者一个内部的状态域或标签。要做到这一点我们必须手动指定这些域的映射。细粒度指定分析器 IK分词器 当我们发送测试单词时默认分词器会将每个字都分开这不符合不符合我们的使用要求所以我们需要下载 ES 对应版本的中文分词器 IK 中文分词器下载网址 将解压后的后的文件夹放入 ES 根目录下的 plugins 目录下重启 ES 即可使用 # GET http://localhost:9200/_analyze {text:测试单词,analyzer:ik_max_word } ik_max_word会将文本做最细粒度的拆分ik_smart会将文本做最粗粒度的拆分 ES 中也可以进行扩展词汇首先查询 #GET http://localhost:9200/_analyze{text:弗雷尔卓德,analyzer:ik_max_word } 首先进入 ES 根目录中的 plugins 文件夹下的 ik 文件夹进入 config 目录创建 custom.dic文件写入弗雷尔卓德同时打开 IKAnalyzer.cfg.xml 文件将新建的 custom.dic 配置其中重启 ES 服务器 ?xml version1.0 encodingUTF-8? !DOCTYPE properties SYSTEM http://java.sun.com/dtd/properties.dtd propertiescommentIK Analyzer 扩展配置/comment!--用户可以在这里配置自己的扩展字典 --entry keyext_dictcustom.dic/entry!--用户可以在这里配置自己的扩展停止词字典--entry keyext_stopwords/entry!--用户可以在这里配置远程扩展字典 --!-- entry keyremote_ext_dictwords_location/entry --!--用户可以在这里配置远程扩展停止词字典--!-- entry keyremote_ext_stopwordswords_location/entry -- /properties 自定义分析器 虽然Elasticsearch带有一些现成的分析器然而在分析器上Elasticsearch真正的强大之处在于你可以通过在一个适合你的特定数据的设置之中组合字符过滤器、分词器、词汇单元过滤器来创建自定义的分析器。在分析与分析器我们说过一个分析器就是在一个包里面组合了三种函数的一个包装器三种函数按照顺序被执行 字符过滤器 字符过滤器用来整理一个尚未被分词的字符串。例如如果我们的文本是HTML格式的它会包含像p或者div这样的HTML标签这些标签是我们不想索引的。我们可以使用html清除字符过滤器来移除掉所有的HTML标签并且像把Aacute;转换为相对应的Unicode字符Á 这样转换HTML实体。一个分析器可能有0个或者多个字符过滤器。 分词器 一个分析器必须有一个唯一的分词器。分词器把字符串分解成单个词条或者词汇单元。标准分析器里使用的标准分词器把一个字符串根据单词边界分解成单个词条并且移除掉大部分的标点符号然而还有其他不同行为的分词器存在。例如关键词分词器完整地输出接收到的同样的字符串并不做任何分词。空格分词器只根据空格分割文本。正则分词器根据匹配正则表达式来分割文本。 词单元过滤器 经过分词作为结果的词单元流会按照指定的顺序通过指定的词单元过滤器。词单元过滤器可以修改、添加或者移除词单元。我们已经提到过lowercase和stop词过滤器但是在Elasticsearch 里面还有很多可供选择的词单元过滤器。词干过滤器把单词遏制为词干。ascii_folding过滤器移除变音符把一个像très”这样的词转换为“tres”。ngram和 edge_ngram词单元过滤器可以产生适合用于部分匹配或者自动补全的词单元。 自定义分析器例子 #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 来 测试这个新的分析器 # GET http://127.0.0.1:9200/my_index/_analyze {text:The quick brown fox,analyzer: my_analyzer } 4.6 文档控制 文档冲突 当我们使用index API更新文档可以一次性读取原始文档做我们的修改然后重新索引整个文档。最近的索引请求将获胜无论最后哪一个文档被索引都将被唯一存储在 Elasticsearch 中。如果其他人同时更改这个文档他们的更改将丢失。 很多时候这是没有问题的。也许我们的主数据存储是一个关系型数据库我们只是将数据复制到Elasticsearch中并使其可被搜索。也许两个人同时更改相同的文档的几率很小。或者对于我们的业务来说偶尔丢失更改并不是很严重的问题。但有时丢失了一个变更就是非常严重的。变更越频繁读数据和更新数据的间隙越长也就越可能丢失变更。在数据库领域中有两种方法通常被用来确保并发更新时变更不会丢失 悲观并发控制这种方法被关系型数据库广泛使用它假定有变更冲突可能发生因此阻塞访问资源以防止冲突。一个典型的例子是读取一行数据之前先将其锁住确保只有放置锁的线程能够对这行数据进行修改。乐观并发控制Elasticsearch 中使用的这种方法假定冲突是不可能发生的并且不会阻塞正在尝试的操作。然而如果源数据在读写当中被修改更新将会失败。应用程序接下来将决定该如何解决冲突。例如可以重试更新、使用新的数据、或者将相关情况报告给用户。 乐观并发控制 Elasticsearch是分布式的。当文档创建、更新或删除时新版本的文档必须复制到集群中的其他节点。Elasticsearch也是异步和并发的这意味着这些复制请求被并行发送并且到达目的地时也许顺序是乱的。Elasticsearch需要一种方法确保文档的旧版本不会覆盖新的版本。 当我们之前讨论index , GET和DELETE请求时我们指出每个文档都有一个_version版本号当文档被修改时版本号递增。Elasticsearch使用这个version号来确保变更以正确顺序得到执行。如果旧版本的文档在新版本之后到达它可以被简单的忽略。我们可以利用version号来确保应用中相互冲突的变更不会导致数据丢失。我们通过指定想要修改文档的 version号来达到这个目的。如果该版本不是当前版本号我们的请求将会失败。老的版本es使用version但是新版本不支持了会报下面的错误提示我们用if_seq _no和if _primary_term # 创建索引数据 PUT http://127.0.0.1:9200/shopping/_create/1001 {title:小米手机,category:小米,images:http://www.gulixueyuan.com/xm.jpg,price:3999.00 } # 返回值 {_index: shopping,_type: _doc,_id: 1001,_version: 1,result: created,_shards: {total: 2,successful: 1,failed: 0},_seq_no: 0,_primary_term: 1 }# 旧版本使用的防止冲突更新方法 POST http://127.0.0.1:9200/shopping/_update/1001?version1 {doc: {title: 小米手机2,category: 小米,images: http://www.gulixueyuan.com/xm.jpg,price: 4999.00} }# 新版本使用的防止冲突更新方法 POST http://127.0.0.1:9200/shopping/_update/1001?if_seq_no1if_primary_term1 {doc: {title: 小米手机2,category: 小米,images: http://www.gulixueyuan.com/xm.jpg,price: 4999.00} } 外部系统版本控制 一个常见的设置是使用其它数据库作为主要的数据存储使用Elasticsearch做数据检索这意味着主数据库的所有更改发生时都需要被复制到Elasticsearch如果多个进程负责这一数据同步你可能遇到类似于之前描述的并发问题。 如果你的主数据库已经有了版本号或一个能作为版本号的字段值比如timestamp那么你就可以在 Elasticsearch 中通过增加 version_typeextermal到查询字符串的方式重用这些相同的版本号版本号必须是大于零的整数且小于9.2E18一个Java中 long类型的正值。外部版本号的处理方式和我们之前讨论的内部版本号的处理方式有些不同Elasticsearch不是检查当前_version和请求中指定的版本号是否相同而是检查当前_version是否小于指定的版本号。如果请求成功外部的版本号作为文档的新_version进行存储。 #POST http://127.0.0.1:9200/shopping/_doc/1001?version100version_typeexternal {title: 小米手机2,category: 小米,images: http://www.gulixueyuan.com/xm.jpg,price: 4999.00 } 4.7 文档展示-Kibana Kibana是一个免费且开放的用户界面能够让你对Elasticsearch 数据进行可视化并让你在Elastic Stack 中进行导航。你可以进行各种操作从跟踪查询负载到理解请求如何流经你的整个应用都能轻松完成。 Kibana下载网址 解压缩下载的 zip 文件修改 config/kibana.yml 文件# 默认端口 server.port: 5601 # ES 服务器的地址 elasticsearch.hosts: [http://localhost:9200] # 索引名 kibana.index: .kibana # 支持中文 i18n.locale: zh-CN Windows 环境下执行 bin/kibana.bat 文件首次启动有点耗时通过浏览器访问http://localhost:5601选择控制台打开可以验证GET shopping/_doc/1001 五、Elasticsearch 集成 1、SpringData 框架集成 1.1 Spring Data 框架使用 Spring Data 的官网 pring Data是一个用于简化数据库、非关系型数据库、索引库访问并支持云服务的开源框架。其主要目标是使得对数据的访问变得方便快捷并支持 map-reduce框架和云计算数据服务。Spring Data可以极大的简化JPA(Elasticsearch…)的写法可以在几乎不用写实现的情况下实现对数据的访问和操作。除了CRUD 外还包括如分页、排序等一些常用的功能。 1.2 Spring Data Elasticsearch 介绍 Spring Data Elasticsearch 官网 Spring Data Elasticsearch基于Spring Data API简化 Elasticsearch 操作将原始操作Elasticsearch 的客户端API进行封装。Spring Data为Elasticsearch 项目提供集成搜索引擎。Spring Data Elasticsearch POJO的关键功能区域为中心的模型与Elastichsearch交互文档和轻松地编写一个存储索引库数据访问层 1.3 框架集成 首先新建项目引入依赖 ?xml version1.0 encodingUTF-8? project xmlnshttp://maven.apache.org/POM/4.0.0xmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsdmodelVersion4.0.0/modelVersiongroupIdcom.example/groupIdartifactIdspringdata-elasticsearch-test/artifactIdversion1.0-SNAPSHOT/versionpropertiesmaven.compiler.source8/maven.compiler.sourcemaven.compiler.target8/maven.compiler.target/propertiesparentgroupIdorg.springframework.boot/groupIdartifactIdspring-boot-starter-parent/artifactIdversion2.3.6.RELEASE/versionrelativePath//parentdependenciesdependencygroupIdorg.projectlombok/groupIdartifactIdlombok/artifactId/dependencydependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-starter-data-elasticsearch/artifactId/dependencydependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-devtools/artifactIdscoperuntime/scopeoptionaltrue/optional/dependencydependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-starter-test/artifactIdscopetest/scope/dependencydependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-test/artifactId/dependencydependencygroupIdjunit/groupIdartifactIdjunit/artifactId/dependencydependencygroupIdorg.springframework/groupIdartifactIdspring-test/artifactId/dependency !-- 自动配置注解处理器及spring-boot-configuration-processor。--dependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-configuration-processor/artifactIdoptionaltrue/optional/dependency/dependencies /project 在 resources 目录中增加application.yml文件 elasticsearch:host: 127.0.0.1 # es 服务地址port: 9200 # es 服务端口# 配置日志级别,开启 debug 日志 logging:level:com.cjz.study: debug 创建数据实体类 Data NoArgsConstructor AllArgsConstructor ToString Document(indexName shopping,shards 3,replicas 1) public class Product{//必须有 id,这里的 id 是全局唯一的标识等同于 es 中的_idIdprivate long id;//商品唯一标识/*** type : 字段数据类型* analyzer : 分词器类型* index : 是否索引(默认:true)* Keyword : 短语,不进行分词*/Field(type FieldType.Text, analyzer ik_max_word)private String title;//商品名称Field(type FieldType.Keyword)private String category;//分类名称 表示该字段内容是一个文本并作为一个整体不可分默认建立索引Field(type FieldType.Double)private Double price;//商品价格Field(type FieldType.Keyword, index false)private String images;//图片地址} 配置类 ElasticsearchRestTemplate是spring-data-elasticsearch项目中的一个类,和其他spring项目中的 template类似。在新版的spring-data-elasticsearch 中ElasticsearchRestTemplate 代替了原来的ElasticsearchTemplate原因是ElasticsearchTemplate基于TransportClientTransportClient即将在8.x 以后的版本中移除。所以我们推荐使用ElasticsearchRestTemplateElasticsearchRestTemplate基于RestHighLevelClient客户端的。需要自定义配置类继承AbstractElasticsearchConfiguration并实现elasticsearchClient()抽象方法创建RestHighLevelClient对象。 public abstract class AbstractElasticsearchConfiguration extends ElasticsearchConfigurationSupport {//需重写本方法public abstract RestHighLevelClient elasticsearchClient();Bean(name { elasticsearchOperations, elasticsearchTemplate })public ElasticsearchOperations elasticsearchOperations(ElasticsearchConverter elasticsearchConverter) {return new ElasticsearchRestTemplate(elasticsearchClient(), elasticsearchConverter);} }需要自定义配置类继承AbstractElasticsearchConfiguration并实现elasticsearchClient()抽象方法创建RestHighLevelClient对象 ConfigurationProperties(prefix elasticsearch) Configuration Data public class ElasticsearchConfig extends AbstractElasticsearchConfiguration {private String host;private Integer port;//重写父类方法Overridepublic RestHighLevelClient elasticsearchClient() {RestClientBuilder builder RestClient.builder(new HttpHost(host, port));return newRestHighLevelClient(builder);} }创建DAO 数据访问对象 Repository public interface ProductDao extends ElasticsearchRepositoryProduct, Long { }1.4 集成测试-索引操作 RunWith(SpringRunner.class) // 选择自己主程序的名字 SpringBootTest(classes Application.class) public class SpringDataESIndexTest {//注入 ElasticsearchRestTemplateAutowiredprivate ElasticsearchRestTemplate elasticsearchRestTemplate;//创建索引并增加映射配置Testpublic void createIndex() {//创建索引系统初始化会自动创建索引System.out.println(创建索引);}//创建索引并增加映射配置Testpublic void deleteIndex() {//创建索引系统初始化会自动创建索引boolean flag elasticsearchRestTemplate.deleteIndex(Product.class);System.out.println(删除索引 flag);}} 1.5 集成测试-文档操作 RunWith(SpringRunner.class) SpringBootTest(classes Application.class) public class SpringDataESProductDaoTest {Autowiredprivate ProductDao productDao;Testpublic void add(){Product product new Product();product.setId(2L);product.setTitle(华为手机);product.setCategory(手机);product.setPrice(2999.0);product.setImages(http://www.atguigu/hw.jpg);productDao.save(product);}//POSTMAN, GET http://localhost:9200/product/_doc/2//修改Testpublic void update(){Product product new Product();product.setId(2L);product.setTitle(小米 2 手机);product.setCategory(手机);product.setPrice(9999.0);product.setImages(http://www.atguigu/xm.jpg);productDao.save(product);}//POSTMAN, GET http://localhost:9200/product/_doc/2Testpublic void getById(){Product product productDao.findById(2L).orElse(new Product());System.out.println(product);}Testpublic void findAll(){IterableProduct products productDao.findAll();for (Product product : products) {System.out.println(product);}}//删除Testpublic void delete(){Product product new Product();product.setId(2L);productDao.delete(product);}//POSTMAN, GET http://localhost:9200/product/_doc/2//批量新增Testpublic void saveAll(){ListProduct productList new ArrayList();for (int i 0; i 10; i) {Product product new Product();product.setId(Long.valueOf(i));product.setTitle([i]小米手机);product.setCategory(手机);product.setPrice(1999.0 i);product.setImages(http://www.atguigu/xm.jpg);productList.add(product);}productDao.saveAll(productList);}Testpublic void findByPageable(){Sort sort Sort.by(Sort.Direction.DESC,id);int currentPage0;//当前页第一页从 0 开始 1 表示第二页int pageSize 5;//每页显示多少条//设置查询分页PageRequest pageRequest PageRequest.of(currentPage, pageSize,sort);//分页查询PageProduct productPage productDao.findAll(pageRequest);for (Product Product : productPage.getContent()) {System.out.println(Product);}} } 1.6 集成测试-文档搜索 RunWith(SpringRunner.class) SpringBootTest(classes Application.class) public class SpringDataESSearchTest {Autowiredprivate ProductDao productDao;Testpublic void termQuery() {TermQueryBuilder queryBuilders QueryBuilders.termQuery(title, 小米);IterableProduct search productDao.search(queryBuilders);for (Product product : search) {System.out.println(product);}}/*** term 查询加分页*/Testpublic void termQueryByPage() {int currentPage 0;int pageSize 5;//设置查询分页PageRequest pageRequest PageRequest.of(currentPage, pageSize);TermQueryBuilder termQueryBuilder QueryBuilders.termQuery(title, 小米);IterableProduct products productDao.search(termQueryBuilder, pageRequest);for (Product product : products) {System.out.println(product);}} } 2、Spark Streaming 框架集成 2.1 Spark Streaming 框架介绍 Spark Streaming 是Spark core API的扩展支持实时数据流的处理并且具有可扩展高吞吐量容错的特点。数据可以从许多来源获取,如Kafka FlumeKinesis或TCP sockets并且可以使用复杂的算法进行处理这些算法使用诸如 mapreducejoin和 window等高级函数表示。最后处理后的数据可以推送到文件系统数据库等。实际上您可以将Spark的机器学习和图形处理算法应用于数据流。 2.2 框架集成 创建新的maven空项目修改pom依赖 ?xml version1.0 encodingUTF-8? project xmlnshttp://maven.apache.org/POM/4.0.0xmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsdparentartifactIdstudy-test/artifactIdgroupIdorg.example/groupIdversion1.0-SNAPSHOT/version/parentmodelVersion4.0.0/modelVersionartifactIdsparkstreaming-elasticsearch/artifactIdpropertiesmaven.compiler.source8/maven.compiler.sourcemaven.compiler.target8/maven.compiler.target/propertiesdependenciesdependencygroupIdorg.apache.spark/groupIdartifactIdspark-core_2.12/artifactIdversion3.0.0/version/dependencydependencygroupIdorg.apache.spark/groupIdartifactIdspark-streaming_2.12/artifactIdversion3.0.0/version/dependencydependencygroupIdorg.elasticsearch/groupIdartifactIdelasticsearch/artifactIdversion7.8.0/version/dependency!-- elasticsearch 的客户端 --dependencygroupIdorg.elasticsearch.client/groupIdartifactIdelasticsearch-rest-high-level-client/artifactIdversion7.8.0/version/dependency!-- elasticsearch 依赖 2.x 的 log4j --dependencygroupIdorg.apache.logging.log4j/groupIdartifactIdlog4j-api/artifactIdversion2.8.2/version/dependencydependencygroupIdorg.apache.logging.log4j/groupIdartifactIdlog4j-core/artifactIdversion2.8.2/version/dependency!-- dependency--!-- groupIdcom.fasterxml.jackson.core/groupId--!-- artifactIdjackson-databind/artifactId--!-- version2.11.1/version--!-- /dependency--!-- lt;!ndash; junit 单元测试 ndash;gt;--!-- dependency--!-- groupIdjunit/groupId--!-- artifactIdjunit/artifactId--!-- version4.12/version--!-- /dependency--/dependencies/project 功能实现 object SparkStreamingESTEST {def main(args: Array[String]): Unit {val sparkConf new SparkConf().setMaster(local[*]).setAppName(ESTest)val ssc new StreamingContext(sparkConf, Seconds(3))val ds: ReceiverInputDStream[String] ssc.socketTextStream(localhost, 9999)ds.foreachRDD(rdd {println(*************** new Date())rdd.foreach(data {val client new RestHighLevelClient(RestClient.builder(new HttpHost(localhost, 9200, http)));// 设置索引及唯一性标识val ss data.split( )// 新增文档 - 请求对象val request new IndexRequest();request.index(product).id(ss(0));val json s| { data : ${ss(1)} }|.stripMargin// 添加文档数据数据格式为 JSON 格式request.source(json,XContentType.JSON);// 客户端发送请求获取响应对象val response client.index(request,RequestOptions.DEFAULT);System.out.println(_index: response.getIndex());System.out.println(_id: response.getId());System.out.println(_result: response.getResult());client.close()})})ssc.start()ssc.awaitTermination();} } 3、Flink 框架集成 3.1 Flink 框架介绍 Apache Spark是一-种基于内存的快速、通用、可扩展的大数据分析计算引擎。Apache Spark掀开了内存计算的先河以内存作为赌注贏得了内存计算的飞速发展。但是在其火热的同时开发人员发现在Spark中计算框架普遍存在的缺点和不足依然没有完全解决而这些问题随着5G时代的来临以及决策者对实时数据分析结果的迫切需要而凸显的更加明显 乱序数据迟到数据低延迟高吞吐准确性容错性数据精准一次性处理Exactly-Once Apache Flink是一个框架和分布式处理引擎用于对无界和有界数据流进行有状态计算。在Spark火热的同时也默默地发展自己并尝试着解决其他计算框架的问题。慢慢地随着这些问题的解决Flink 慢慢被绝大数程序员所熟知并进行大力推广阿里公司在2015年改进Flink并创建了内部分支Blink目前服务于阿里集团内部搜索、推荐、广告和蚂蚁等大量核心实时业务 3.2 demo实现 首先创建空项目引入依赖 ?xml version1.0 encodingUTF-8? project xmlnshttp://maven.apache.org/POM/4.0.0xmlns:xsihttp://www.w3.org/2001/XMLSchema-instancexsi:schemaLocationhttp://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsdparentartifactIdstudy-test/artifactIdgroupIdorg.example/groupIdversion1.0-SNAPSHOT/version/parentmodelVersion4.0.0/modelVersionartifactIdflink-elasticsearch/artifactIdpropertiesmaven.compiler.source8/maven.compiler.sourcemaven.compiler.target8/maven.compiler.target/propertiesdependenciesdependencygroupIdorg.apache.flink/groupIdartifactIdflink-scala_2.12/artifactIdversion1.12.0/version/dependencydependencygroupIdorg.apache.flink/groupIdartifactIdflink-streaming-scala_2.12/artifactIdversion1.12.0/version/dependencydependencygroupIdorg.apache.flink/groupIdartifactIdflink-clients_2.12/artifactIdversion1.12.0/version/dependencydependencygroupIdorg.apache.flink/groupIdartifactIdflink-connector-elasticsearch7_2.11/artifactIdversion1.12.0/version/dependency!-- jackson --dependencygroupIdcom.fasterxml.jackson.core/groupIdartifactIdjackson-core/artifactIdversion2.11.1/version/dependencydependencygroupIdcommons-logging/groupIdartifactIdcommons-logging/artifactIdversion1.2/version/dependency/dependencies/project 功能实现 public class FlinkElasticsearchSinkTest {public static void main(String[] args) throws Exception {//构建Flink 环境对象StreamExecutionEnvironment env StreamExecutionEnvironment.getExecutionEnvironment();//数据的输入DataStreamSourceString source env.socketTextStream(localhost, 9999);//使用ESBuilder 构建输出ListHttpHost hostList new ArrayList();hostList.add(new HttpHost(127.0.0.1, 9200, http));ElasticsearchSink.BuilderString stringBuilder new ElasticsearchSink.Builder(hostList,new ElasticsearchSinkFunctionString() {Overridepublic void process(String s, RuntimeContext runtimeContext, RequestIndexer requestIndexer) {IndexRequest indexRequest Requests.indexRequest();MapString, String json new HashMap();json.put(data, s);indexRequest.index(flink-index);indexRequest.id(9001);indexRequest.source(json);requestIndexer.add(indexRequest);}});//Sink 数据的输出stringBuilder.setBulkFlushMaxActions(1);source.addSink(stringBuilder.build());//执行操作env.execute(flink-job);} } 六、Elasticsearch 优化 1、硬件选择 Elasticsearch 的基础是 Lucene所有的索引和文档数据是存储在本地的磁盘中具体的路径可在 ES 的配置文件…/config/elasticsearch.yml中配置如下 # # Path to directory where to store the data (separate multiple locations by comma): # path.data: /path/to/data # # Path to log files: # path.logs: /path/to/logs 使用SSD就像其他地方提过的他们比机械磁盘优秀多了使用RAID0。条带化RAID会提高磁盘IO代价显然就是当一块硬盘故障时整个就故障了。不要使用镜像或者奇偶校验RAID因为副本已经提供了这个功能另外使用多块硬盘并允许Elasticsearch 通过多个path data目录配置把数据条带化分配到它们上面不要使用远程挂载的存储比如NFS或者SMB/CIFS。这个引入的延迟对性能来说完全是背道而驰的 2、分片策略 2.1 合理设置分片数 分片和副本的设计为 ES 提供了支持分布式和故障转移的特性但并不意味着分片和副本是可以无限分配的。而且索引的分片完成分配后由于索引的路由机制我们是不能重新修改分片数的。但是需要知道的是一个分片并不是没有代价的 一个分片的底层即为一个 Lucene 索引会消耗一定文件句柄、内存、以及 CPU运转每一个搜索请求都需要命中索引中的每一个分片如果每一个分片都处于不同的节点还好 但如果多个分片都需要在同一个节点上竞争使用相同的资源就有些糟糕了用于计算相关度的词项统计信息是基于分片的。如果有许多分片每一个都只有很少的数据会导致很低的相关度 一个业务索引具体需要分配多少分片可能需要架构师和技术人员对业务的增长有个预先的判断横向扩展应当分阶段进行。为下一阶段准备好足够的资源。 只有当你进入到下一个阶段你才有时间思考需要作出哪些改变来达到这个阶段。一般来说我们遵循一些原则 控制每个分片占用的硬盘容量不超过 ES 的最大 JVM 的堆空间设置一般设置不超过 32G参考下文的 JVM 设置原则因此如果索引的总容量在 500G 左右那分片大小在 16 个左右即可当然最好同时考虑原则 2。 考虑一下 node 数量一般一个节点有时候就是一台物理机如果分片数过多大大超过了节点数很可能会导致一个节点上存在多个分片一旦该节点故障即使保持了 1 个以上的副本同样有可能会导致数据丢失集群无法恢复。所以 一般都设置分片数不超过节点数的 3 倍。 主分片副本和节点最大数之间数量我们分配的时候可以参考以下关系 节点数主分片数 *副本数1 2.2 推迟分片分配 对于节点瞬时中断的问题默认情况集群会等待一分钟来查看节点是否会重新加入如果这个节点在此期间重新加入重新加入的节点会保持其现有的分片数据不会触发新的分片分配。这样就可以减少 ES 在自动再平衡可用分片时所带来的极大开销。通过修改参数 delayed_timeout 可以延长再均衡的时间可以全局设置也可以在索引级别进行修改 #PUT /_all/_settings {settings: {index.unassigned.node_left.delayed_timeout: 5m} } 3、路由选择 当我们查询文档的时候 Elasticsearch 如何知道一个文档应该存放到哪个分片中呢它其实是通过下面这个公式来计算出来 shard hash(routing) % number_of_primary_shards routing 默认值是文档的 id也可以采用自定义值比如用户 id。 不带routing查询 在查询的时候因为不知道要查询的数据具体在哪个分片上所以整个过程分为2个步骤 分发请求到达协调节点后协调节点将查询请求分发到每个分片上。聚合协调节点搜集到每个分片上查询结果在将查询的结果进行排序之后给用户返回结果。 带routing查询 询的时候可以直接根据routing 信息定位到某个分配查询不需要查询所有的分配经过协调节点排序。向上面自定义的用户查询如果routing 设置为userid 的话就可以直接查询出数据来效率提升很多。 4、写入速度优化 ES 的默认配置是综合了数据可靠性、写入速度、搜索实时性等因素。实际使用时我们需要根据公司要求进行偏向性的优化。针对于搜索性能要求不高但是对写入要求较高的场景我们需要尽可能的选择恰当写优化策略。综合来说可以考虑以下几个方面来提升写索引的性能 加大Translog Flush目的是降低Iops、Writeblock增加Index Refesh间隔目的是减少Segment Merge的次数调整Bulk 线程池和队列优化节点间的任务分布优化Lucene层的索引建立目的是降低CPU及IO 4.1 优化存储设备 ES 是一种密集使用磁盘的应用在段合并的时候会频繁操作磁盘所以对磁盘要求较高当磁盘速度提升之后集群的整体性能会大幅度提高 4.2 合理使用合并 Lucene 以段的形式存储数据。当有新的数据写入索引时 Lucene 就会自动创建一个新的段。随着数据量的变化段的数量会越来越多消耗的多文件句柄数及 CPU 就越多查询效率就会下降。由于 Lucene 段合并的计算量庞大会消耗大量的 I/O所以 ES 默认采用较保守的策略让后台定期进行段合并。 4.3 减少 Refresh 的次数 Lucene 在新增数据时采用了延迟写入的策略默认情况下索引的refresh_interval 为1 秒。Lucene 将待写入的数据先写到内存中超过 1 秒默认时就会触发一次 Refresh然后 Refresh 会把内存中的的数据刷新到操作系统的文件缓存系统中。如果我们对搜索的实效性要求不高可以将 Refresh 周期延长例如 30 秒。这样还可以有效地减少段刷新次数但这同时意味着需要消耗更多的 Heap 内存。 4.4 加大 Flush 设置 Flush 的主要目的是把文件缓存系统中的段持久化到硬盘当 Translog 的数据量达到 512MB 或者 30 分钟时会触发一次 Flush。index.translog.flush_threshold_size 参数的默认值是 512MB我们进行修改。增加参数值意味着文件缓存系统中可能需要存储更多的数据所以我们需要为操作系统的文件缓存系统留下足够的空间。 4.5 减少副本的数量 ES 为了保证集群的可用性提供了 Replicas副本支持然而每个副本也会执行分析、索引及可能的合并过程所以 Replicas 的数量会严重影响写索引的效率。 当写索引时需要把写入的数据都同步到副本节点副本节点越多写索引的效率就越慢。如果我们需要大批量进行写入操作可以先禁止Replica复制设置 index.number_of_replicas: 0 关闭副本。在写入完成后 Replica 修改回正常的状态。 5、内存设置 ES 默认安装后设置的内存是 1GB对于任何一个现实业务来说这个设置都太小了。如果是通过解压安装的 ES则在 ES 安装文件中包含一个 jvm.option 文件添加如下命令来设置 ES 的堆大小 Xms 表示堆的初始大小 Xmx 表示可分配的最大内存都是 1GB。确保 Xmx 和 Xms 的大小是相同的其目的是为了能够在 Java 垃圾回收机制清理完堆区后不需要重新分隔计算堆区的大小而浪费资源可以减轻伸缩堆大小带来的压力。 假设你有一个 64G 内存的机器按照正常思维思考你可能会认为把 64G 内存都给ES 比较好但现实是这样吗 越大越好虽然内存对 ES 来说是非常重要的但是答案是否定的因为 ES 堆内存的分配需要满足以下两个原则 不要超过物理内存的 50% Lucene 的设计目的是把底层 OS 里的数据缓存到内存中。Lucene 的段是分别存储到单个文件中的这些文件都是不会变化的所以很利于缓存同时操作系统也会把这些段文件缓存起来以便更快的访问。如果我们设置的堆内存过大 Lucene 可用的内存将会减少就会严重影响降低 Lucene 的全文本查询性能。堆内存的大小最好不要超过 32GB在 Java 中所有对象都分配在堆上然后有一个 Klass Pointer 指针指向它的类元数据。这个指针在 64 位的操作系统上为 64 位 64 位的操作系统可以使用更多的内存2^64。在 32 位的系统上为 32 位 32 位的操作系统的最大寻址空间为 4GB2^32。但是 64 位的指针意味着更大的浪费因为你的指针本身大了。浪费内存不算更糟糕的是更大的指针在主内存和缓存器例如 LLC, L1 等之间移动数据的时候会占用更多的带宽。 最终我们都会采用 31 G 设置 -Xms 31g-Xmx 31g 假设你有个机器有 128 GB 的内存你可以创建两个节点每个节点内存分配不超过 32 GB。也就是说不超过 64 GB 内存给 ES 的堆内存剩下的超过 64 GB 的内存给 Lucene 6、重要配置项 参数名**参数值 **说明cluster.nameelasticsearch配置 ES 的集群名称默认值是 ES建议改成与所存数据相关的名称 ES 会自动发现在同一网段下的 集群名称相同的节点node.namenode-1集群中的节点名在同一个集群中不能重复。节点 的名称一旦设置就不能再改变了。当然也可以 设 置 成 服 务 器 的 主 机 名 称 例 如 node.name:${HOSTNAME}node.mastertrue指定该节点是否有资格被选举成为 Master 节点默 认是 True如果被设置为 True则只是有资格成为 Master 节点具体能否成为 Master 节点需要通 过选举产生node.datatrue指定该节点是否存储索引数据默认为 True。数据 的增、删、改、查都是在 Data 节点完成的index.number_of_shards1设置都索引分片个数默认是 1 片。也可以在创建 索引时设置该值具体设置为多大都值要根据数据 量的大小来定。如果数据量不大则设置成 1 时效 率最高index.number_of_replicas1设置默认的索引副本个数默认为 1 个。副本数越 多集群的可用性越好但是写索引时需要同步的 数据越多transport.tcp.compresstrue设置在节点间传输数据时是否压缩默认为 False 不压缩discovery.zen.minimum_master_nodes1设置在选举 Master 节点时需要参与的最少的候选 主节点数默认为 1。如果使用默认值则当网络 不稳定时有可能会出现脑裂。 合 理 的 数 值 为 (master_eligible_nodes/2)1 其 中 master_eligible_nodes 表示集群中的候选主节点数discovery.zen.ping.timeout3s设置在集群中自动发现其他节点时 Ping 连接的超 时时间默认为 3 秒。 在较差的网络环境下需要设置得大一点防止因误 判该节点的存活状态而导致分片的转移 七、Elasticsearch面试题 1、为什么要使用 Elasticsearch 系统中的数据 随着业务的发展时间的推移 将会非常多 而业务中往往采用模糊查询进行数据的搜索 而模糊查询会导致查询引擎放弃索引导致系统查询数据时都是全表扫描在百万级别的数据库中查询效率是非常低下的而我们使用 ES 做一个全文索引将经常查询的系统功能的某些字段比如说电商系统的商品表中商品名描述、价格还有 id 这些字段我们放入 ES 索引库里可以提高查询速度。 2、Elasticsearch 的 master 选举流程 Elasticsearch的选主是ZenDiscovery模块负责的主要包含Ping节点之间通过这个RPC来发现彼此和Unicast单播模块包含-一个主机列表以控制哪些节点需要ping通这两部分对所有可以成为master的节点node master: true根据nodeId字典排序每次选举每个节点都把自己所知道节点排一次序然后选出第一个第0位节点暂且认为它是master节点如果对某个节点的投票数达到一定的值可以成为master节点数n/21并且该节点自己也选举自己那这个节点就是master。否则重新选举一直到满足上述条件master节点的职责主要包括集群、节点和索引的管理不负责文档级别的管理data节点可以关闭http功能 3、Elasticsearch 集群脑裂问题 脑裂问题可能的成因 网络问题集群间的网络延迟导致一些节点访问不到master, 认为master 挂掉了从而选举出新的master,并对master上的分片和副本标红分配新的主分片节点负载主节点的角色既为master又为data,访问量较大时可能会导致ES停止响应造成大面积延迟此时其他节点得不到主节点的响应认为主节点挂掉了会重新选取主节点内存回收data 节点上的ES进程占用的内存较大引发JVM的大规模内存回收造成ES进程失去响应 脑裂问题解决方案 减少误判discovery.zen.ping_timeout 节点状态的响应时间默认为3s可以适当调大如果master在该响应时间的范围内没有做出响应应答判断该节点已经挂掉了。调大参数如6sdiscovery.zen.ping_timeout:6可适当减少误判选举触发discovery.zen.minimum_master_nodes:1该参數是用于控制选举行为发生的最小集群主节点数量。当备选主节点的个數大于等于该参数的值且备选主节点中有该参数个节点认为主节点挂了进行选举。官方建议为(n / 2) 1, n为主节点个数即有资格成为主节点的节点个数角色分离即master节点与data节点分离限制角色 主节点配置为node.master: truenode.data: false从节点配置为node.master: falsenode.data: true 4、Elasticsearch 索引文档的流程 协调节点默认使用文档 ID 参与计算也支持通过 routing以便为路由提供合适的分片shard hash(document_id) % (num_of_primary_shards)当分片所在的节点接收到来自协调节点的请求后会将请求写入到 Memory Buffer然后定时默认是每隔 1 秒写入到 Filesystem Cache这个从 Memory Buffer 到 Filesystem Cache 的过程就叫做 refresh当然在某些情况下存在 Momery Buffer 和 Filesystem Cache 的数据可能会丢失 ES 是通过 translog的机制来保证数据的可靠性的。其实现机制是接收到请求后同时也会写入到 translog 中当 Filesystemcache 中的数据写入到磁盘中时才会清除掉这个过程叫做 flush在 flush 过程中内存中的缓冲将被清除内容被写入一个新段段的 fsync 将创建一个新的提交点并将内容刷新到磁盘旧的 translog 将被删除并开始一个新的 translog。flush 触发的时机是定时触发默认 30 分钟或者 translog 变得太大默认为 512M时 5、Elasticsearch 更新和删除文档的流程 删除和更新也都是写操作但是 Elasticsearch 中的文档是不可变的因此不能被删除或者改动以展示其变更磁盘上的每个段都有一个相应的.del 文件。当删除请求发送后文档并没有真的被删除而是在.del文件中被标记为删除。该文档依然能匹配查询但是会在结果中被过滤掉。当段合并时在.del 文件中被标记为删除的文档将不会被写入新段。在新的文档被创建时 Elasticsearch 会为该文档指定一个版本号当执行更新时旧版本的文档在.del文件中被标记为删除新版本的文档被索引到一个新段。旧版本的文档依然能匹配查询但是会在结果中被过滤掉。 6、Elasticsearch 搜索的流程 搜索被执行成一个两阶段过程我们称之为 Query Then Fetch在初始查询阶段时查询会广播到索引中每一个分片拷贝主分片或者副本分片。 每个分片在本地执行搜索并构建一个匹配文档的大小为 from size 的优先队列。 PS在搜索的时候是会查询Filesystem Cache 的但是有部分数据还在 Memory Buffer所以搜索是近实时的。每个分片返回各自优先队列中 所有文档的 ID 和排序值 给协调节点它合并这些值到自己的优先队列中来产生一个全局排序后的结果列表。接下来就是取回阶段 协调节点辨别出哪些文档需要被取回并向相关的分片提交多个 GET 请求。每个分片加载并丰富文档如果有需要的话接着返回文档给协调节点。一旦所有的文档都被取回了协调节点返回结果给客户端。Query Then Fetch 的搜索类型在文档相关性打分的时候参考的是本分片的数据这样在文档数量较少的时候可能不够准确 DFS Query Then Fetch 增加了一个预查询的处理询问 Term 和 Document frequency这个评分更准确但是性能会变差。 7、Elasticsearch 在部署时对 Linux 的设置有哪些优化方法 64 GB 内存的机器是非常理想的 但是 32 GB 和 16 GB 机器也是很常见的。少于 8 GB 会适得其反如果你要在更快的 CPUs 和更多的核心之间选择选择更多的核心更好。多个内核提供的额外并发远胜过稍微快一点点的时钟频率如果你负担得起 SSD它将远远超出任何旋转介质。 基于 SSD 的节点查询和索引性能都有提升。如果你负担得起 SSD 是一个好的选择即使数据中心们近在咫尺也要避免集群跨越多个数据中心。绝对要避免集群跨越大的地理距离请确保运行你应用程序的 JVM 和服务器的 JVM 是完全一样的。 在 Elasticsearch 的几个地方使用 Java 的本地序列化通过设置 gateway.recover_after_nodes、 gateway.expected_nodes、 gateway.recover_after_time 可以在集群重启的时候避免过多的分片交换这可能会让数据恢复从数个小时缩短为几秒钟Elasticsearch 默认被配置为使用单播发现以防止节点无意中加入集群。只有在同一台机器上运行的节点才会自动组成集群。最好使用单播代替组播不要随意修改垃圾回收器CMS和各个线程池的大小把你的内存的少于一半给 Lucene但不要超过 32 GB通过 ES_HEAP_SIZE 环境变量设置内存交换到磁盘对服务器性能来说是致命的。如果内存交换到磁盘上一个 100 微秒的操作可能变成 10 毫秒。 再想想那么多 10 微秒的操作时延累加起来。 不难看出 swapping 对于性能是多么可怕Lucene 使用了大量的文件。同时 Elasticsearch 在节点和 HTTP 客户端之间进行通信也使用了大量的套接字。 所有这一切都需要足够的文件描述符。你应该增加你的文件描述符设置一个很大的值如 64,000 8、GC 方面在使用 Elasticsearch 时要注意什么 倒排词典的索引需要常驻内存无法 GC需要监控 data node 上 segment memory 增长趋势。各类缓存 field cache, filter cache, indexing cache, bulk queue 等等要设置合理的大小并且要应该根据最坏的情况来看 heap 是否够用也就是各类缓存全部占满的时候还有 heap 空间可以分配给其他任务吗避免采用 clear cache 等“自欺欺人”的方式来释放内存避免返回大量结果集的搜索与聚合。确实需要大量拉取数据的场景可以采用 scan scroll api 来实现cluster stats 驻留内存并无法水平扩展超大规模集群可以考虑分拆成多个集群通过 tribe node 连接想知道 heap 够不够必须结合实际应用场景并对集群的 heap 使用情况做持续的监控 9、Elasticsearch 对于大数据量上亿量级的聚合如何实现 Elasticsearch 提供的首个近似聚合是 cardinality 度量。它提供一个字段的基数即该字段的 distinct或者 unique 值的数目。它是基于 HLL 算法的。 HLL 会先对我们的输入作哈希运算然后根据哈希运算的结果中的 bits 做概率估算从而得到基数。其特点是可配置的精度用来控制内存的使用更精确 更多内存小的数据集精度是非常高的我们可以通过配置参数来设置去重需要的固定内存使用量。无论数千还是数十亿的唯一值内存使用量只与你配置的精确度相关。 10、在并发情况下 Elasticsearch 如果保证读写一致 可以通过版本号使用乐观并发控制以确保新版本不会被旧版本覆盖由应用层来处理具体的冲突另外对于写操作一致性级别支持 quorum/one/all默认为 quorum即只有当大多数分片可用时才允许写操作。但即使大多数可用也可能存在因为网络等原因导致写入副本失败这样该副本被认为故障分片将会在一个不同的节点上重建。对于读操作可以设置** replication 为 sync(默认)这使得操作在主分片和副本分片都完成后才会返回如果设置 replication 为 async 时也可以通过设置搜索请求参数_preference 为 primary** 来查询主分片确保文档是最新版本。 11、如何监控 Elasticsearch 集群状态 elasticsearch-head 插件。通过 Kibana 监控 Elasticsearch。你可以实时查看你的集群健康状态和性能也可以分析过去的集群、索引和节点指标 12、是否了解字典树 字典树又称单词查找树 Trie 树是一种树形结构是一种哈希树的变种。典型应用是用于统计排序和保存大量的字符串但不仅限于字符串所以经常被搜索引擎系统用于文本词频统计。它的优点是利用字符串的公共前缀来减少查询时间最大限度地减少无谓的字符串比较查询效率比哈希树高。 Trie 的核心思想是空间换时间利用字符串的公共前缀来降低查询时间的开销以达到提高效率的目的。它有 3 个基本性质 根节点不包含字符除根节点外每一个节点都只包含一个字符。从根节点到某一节点路径上经过的字符连接起来为该节点对应的字符串。每个节点的所有子节点包含的字符都不相同 对于中文的字典树每个节点的子节点用一个哈希表存储这样就不用浪费太大的空间而且查询速度上可以保留哈希的复杂度 O(1) 13、Elasticsearch 中的集群、节点、索引、文档、类型是什么 集群是一个或多个节点服务器的集合它们共同保存您的整个数据并提供跨所有节点的联合索引和搜索功能。群集由唯一名 称标识默认情况下为elasticsearch。此名称很重要因为如果节点设置为按名称加入群集则该节点只能是群集的一部分节点是属于集群一部分的单个服务器。它存储数据并参与群集索引和搜索功能。索引就像关系数据库中的“数据库”。它有一个定义多种类型的映射。索引是逻辑名称空间映射到一个或多个主分片并且可以有零个或多个副本分片。MySQL 数据库Elasticsearch索引文档类似于关系数据库中的一行。不同之处在于索引中的每个文档可以具有不同的结构(字段)但是对于通用字段应该具有相同的数据类型。MySQL Databases Tables Columns / RowsElasticsearch Indices Types 具有属性的文档Doc类型是索引的逻辑类别/分区其语义完全取决于用户 14、Elasticsearch 中的倒排索引是什么 倒排索引是搜索引擎的核心。搜索引擎的主要目标是在查找发生搜索条件的文档时提供快速搜索。ES中的倒排索引其实就是 lucene 的倒排索引区别于传统的正向索引 倒排索引会再存储数据时将关键词和数据进行关联保存到倒排表中然后查询时将查询内容进行分词后在倒排表中进行查询最后匹配数据即可 参考资料https://www.bilibili.com/video/BV1hh411D7sb/
http://www.lakalapos1.cn/news/48017/

相关文章:

  • 桂林网站建设凡森网络网站搭建费用明细
  • 极简风格 网站市场营销策略分析案例
  • 西安建站之家网络科技有限公司广州最大的跨境电商公司排名
  • 广东企业网站建设多少钱服务器调用wordpress
  • 做老师讲课视频的教育网站网站总体设计方案
  • 湖北省建设工程质量安全监督网站有没有做网站的团队
  • 网站建设方案报价费用明细价格山西响应式网站哪家好
  • 网站开发 工期安排广州网站排名优化
  • 安徽省途顺建设工程有限公司网站石家网站建设公司排名
  • 网站建设中应该返回502还是301邢台贴吧最新消息
  • 工商企业网站肇庆城乡建设门户网站
  • 网站建设费预付定金什么科目怎样打造营销型网站建设
  • 网站维护主要工作内容网页设计主要学什么内容
  • 精品网站网站登录模版
  • 茂名做网站阿里巴巴国际网站怎么做
  • 网站模块源码网站如何做超级链接
  • 网站首页开发网站优化反馈机制 seo
  • 网站建设公司一年多少钱小程序在哪个网站做
  • 梧州网站推广方案企业网站建设中在方案设计上
  • 赣州门户网站建设国内有做网游评测的网站么
  • 专门做算法项目的网站做网站源码需要多少钱
  • 做设计什么兼职网站买网站源码的网站
  • 减肥网站开发目的在线优化工具
  • 帮做毕设的网站找不同 网站开发
  • 做网站关键词必须要中文网址常见的域名
  • next.js做纯静态网站wordpress 所有页面空白
  • 一个完整的网站设计需要的技术免费网站建设网站有那些
  • 1688货源网聊城哪里做优化网站
  • 网站标题设计在线网站宣传海报图片
  • 宜宾建功路桥建设有限公司网站网站建设客户说没用