一、高可用简介

Hadoop 高可用 (High Availability) 分为 HDFS 高可用和 YARN 高可用,两者的实现基本类似,但 HDFS NameNode 对数据存储及其一致性的要求比 YARN ResourceManger 高得多,所以它的实现也更加复杂,故下面先进行讲解:

1.1 高可用整体架构

HDFS 高可用架构如下:

图片引用自:https://www.edureka.co/blog/how-to-set-up-hadoop-cluster-with-hdfs-high-availability/

HDFS 高可用架构主要由以下组件所构成:

  • Active NameNode 和 Standby NameNode:两台 NameNode 形成互备,一台处于 Active 状态,为主 NameNode,另外一台处于 Standby 状态,为备 NameNode,只有主 NameNode 才能对外提供读写服务。
  • 主备切换控制器 ZKFailoverController:ZKFailoverController 作为独立的进程运行,对 NameNode 的主备切换进行总体控制。ZKFailoverController 能及时检测到 NameNode 的健康状况,在主 NameNode 故障时借助 Zookeeper 实现自动的主备选举和切换,当然 NameNode 目前也支持不依赖于 Zookeeper 的手动主备切换。
  • Zookeeper 集群:为主备切换控制器提供主备选举支持。
  • 共享存储系统:共享存储系统是实现 NameNode 的高可用最为关键的部分,共享存储系统保存了 NameNode 在运行过程中所产生的 HDFS 的元数据。主 NameNode 和 NameNode 通过共享存储系统实现元数据同步。在进行主备切换的时候,新的主 NameNode 在确认元数据完全同步之后才能继续对外提供服务。
  • DataNode 节点:除了通过共享存储系统共享 HDFS 的元数据信息之外,主 NameNode 和备 NameNode 还需要共享 HDFS 的数据块和 DataNode 之间的映射关系。DataNode 会同时向主 NameNode 和备 NameNode 上报数据块的位置信息。

1.2 基于 QJM 的共享存储系统的数据同步机制分析

目前 Hadoop 支持使用 Quorum Journal Manager (QJM) 或 Network File System (NFS) 作为共享的存储系统,这里以 QJM 集群为例进行说明:Active NameNode 首先把 EditLog 提交到 JournalNode 集群,然后 Standby NameNode 再从 JournalNode 集群定时同步 EditLog,当 Active NameNode 宕机后, Standby NameNode 在确认元数据完全同步之后就可以对外提供服务。

需要说明的是向 JournalNode 集群写入 EditLog 是遵循 “过半写入则成功” 的策略,所以你至少要有3个 JournalNode 节点,当然你也可以继续增加节点数量,但是应该保证节点总数是奇数。同时如果有 2N+1 台 JournalNode,那么根据过半写的原则,最多可以容忍有 N 台 JournalNode 节点挂掉。

1.3 NameNode 主备切换

NameNode 实现主备切换的流程下图所示:

  1. HealthMonitor 初始化完成之后会启动内部的线程来定时调用对应 NameNode 的 HAServiceProtocol RPC 接口的方法,对 NameNode 的健康状态进行检测。

  2. HealthMonitor 如果检测到 NameNode 的健康状态发生变化,会回调 ZKFailoverController 注册的相应方法进行处理。

  3. 如果 ZKFailoverController 判断需要进行主备切换,会首先使用 ActiveStandbyElector 来进行自动的主备选举。

  4. ActiveStandbyElector 与 Zookeeper 进行交互完成自动的主备选举。

  5. ActiveStandbyElector 在主备选举完成后,会回调 ZKFailoverController 的相应方法来通知当前的 NameNode 成为主 NameNode 或备 NameNode。

  6. ZKFailoverController 调用对应 NameNode 的 HAServiceProtocol RPC 接口的方法将 NameNode 转换为 Active 状态或 Standby 状态。

1.4 YARN高可用

YARN ResourceManager 的高可用与 HDFS NameNode 的高可用类似,但是 ResourceManager 不像 NameNode ,没有那么多的元数据信息需要维护,所以它的状态信息可以直接写到 Zookeeper 上,并依赖 Zookeeper 来进行主备选举。

二、集群规划

按照高可用的设计目标:需要保证至少有两个 NameNode (一主一备) 和 两个 ResourceManager (一主一备) ,同时为满足“过半写入则成功”的原则,需要至少要有3个 JournalNode 节点。这里使用三台主机进行搭建,集群规划如下:

三、前置条件

四、集群配置

4.1 下载并解压

下载Hadoop。这里我下载的是CDH版本Hadoop,下载地址为:http://archive.cloudera.com/cdh5/cdh/5/

  1. # tar -zvxf hadoop-2.6.0-cdh5.15.2.tar.gz

4.2 配置环境变量

编辑profile文件:

  1. # vim /etc/profile

增加如下配置:

  1. export HADOOP_HOME=/usr/app/hadoop-2.6.0-cdh5.15.2
  2. export PATH=${HADOOP_HOME}/bin:$PATH

执行source命令,使得配置立即生效:

  1. # source /etc/profile

4.3 修改配置

进入${HADOOP_HOME}/etc/hadoop目录下,修改配置文件。各个配置文件内容如下:

1. hadoop-env.sh

  1. # 指定JDK的安装位置
  2. export JAVA_HOME=/usr/java/jdk1.8.0_201/

2. core-site.xml

  1. <configuration>
  2. <property>
  3. <!-- 指定namenode的hdfs协议文件系统的通信地址 -->
  4. <name>fs.defaultFS</name>
  5. <value>hdfs://hadoop001:8020</value>
  6. </property>
  7. <property>
  8. <!-- 指定hadoop集群存储临时文件的目录 -->
  9. <name>hadoop.tmp.dir</name>
  10. <value>/home/hadoop/tmp</value>
  11. </property>
  12. <property>
  13. <!-- ZooKeeper集群的地址 -->
  14. <name>ha.zookeeper.quorum</name>
  15. <value>hadoop001:2181,hadoop002:2181,hadoop002:2181</value>
  16. </property>
  17. <property>
  18. <!-- ZKFC连接到ZooKeeper超时时长 -->
  19. <name>ha.zookeeper.session-timeout.ms</name>
  20. <value>10000</value>
  21. </property>
  22. </configuration>

3. hdfs-site.xml

  1. <configuration>
  2. <property>
  3. <!-- 指定HDFS副本的数量 -->
  4. <name>dfs.replication</name>
  5. <value>3</value>
  6. </property>
  7. <property>
  8. <!-- namenode节点数据(即元数据)的存放位置,可以指定多个目录实现容错,多个目录用逗号分隔 -->
  9. <name>dfs.namenode.name.dir</name>
  10. <value>/home/hadoop/namenode/data</value>
  11. </property>
  12. <property>
  13. <!-- datanode节点数据(即数据块)的存放位置 -->
  14. <name>dfs.datanode.data.dir</name>
  15. <value>/home/hadoop/datanode/data</value>
  16. </property>
  17. <property>
  18. <!-- 集群服务的逻辑名称 -->
  19. <name>dfs.nameservices</name>
  20. <value>mycluster</value>
  21. </property>
  22. <property>
  23. <!-- NameNode ID列表-->
  24. <name>dfs.ha.namenodes.mycluster</name>
  25. <value>nn1,nn2</value>
  26. </property>
  27. <property>
  28. <!-- nn1的RPC通信地址 -->
  29. <name>dfs.namenode.rpc-address.mycluster.nn1</name>
  30. <value>hadoop001:8020</value>
  31. </property>
  32. <property>
  33. <!-- nn2的RPC通信地址 -->
  34. <name>dfs.namenode.rpc-address.mycluster.nn2</name>
  35. <value>hadoop002:8020</value>
  36. </property>
  37. <property>
  38. <!-- nn1的http通信地址 -->
  39. <name>dfs.namenode.http-address.mycluster.nn1</name>
  40. <value>hadoop001:50070</value>
  41. </property>
  42. <property>
  43. <!-- nn2的http通信地址 -->
  44. <name>dfs.namenode.http-address.mycluster.nn2</name>
  45. <value>hadoop002:50070</value>
  46. </property>
  47. <property>
  48. <!-- NameNode元数据在JournalNode上的共享存储目录 -->
  49. <name>dfs.namenode.shared.edits.dir</name>
  50. <value>qjournal://hadoop001:8485;hadoop002:8485;hadoop003:8485/mycluster</value>
  51. </property>
  52. <property>
  53. <!-- Journal Edit Files的存储目录 -->
  54. <name>dfs.journalnode.edits.dir</name>
  55. <value>/home/hadoop/journalnode/data</value>
  56. </property>
  57. <property>
  58. <!-- 配置隔离机制,确保在任何给定时间只有一个NameNode处于活动状态 -->
  59. <name>dfs.ha.fencing.methods</name>
  60. <value>sshfence</value>
  61. </property>
  62. <property>
  63. <!-- 使用sshfence机制时需要ssh免密登录 -->
  64. <name>dfs.ha.fencing.ssh.private-key-files</name>
  65. <value>/root/.ssh/id_rsa</value>
  66. </property>
  67. <property>
  68. <!-- SSH超时时间 -->
  69. <name>dfs.ha.fencing.ssh.connect-timeout</name>
  70. <value>30000</value>
  71. </property>
  72. <property>
  73. <!-- 访问代理类,用于确定当前处于Active状态的NameNode -->
  74. <name>dfs.client.failover.proxy.provider.mycluster</name>
  75. <value>org.apache.hadoop.hdfs.server.namenode.ha.ConfiguredFailoverProxyProvider</value>
  76. </property>
  77. <property>
  78. <!-- 开启故障自动转移 -->
  79. <name>dfs.ha.automatic-failover.enabled</name>
  80. <value>true</value>
  81. </property>
  82. </configuration>

4. yarn-site.xml

  1. <configuration>
  2. <property>
  3. <!--配置NodeManager上运行的附属服务。需要配置成mapreduce_shuffle后才可以在Yarn上运行MapReduce程序。-->
  4. <name>yarn.nodemanager.aux-services</name>
  5. <value>mapreduce_shuffle</value>
  6. </property>
  7. <property>
  8. <!-- 是否启用日志聚合(可选) -->
  9. <name>yarn.log-aggregation-enable</name>
  10. <value>true</value>
  11. </property>
  12. <property>
  13. <!-- 聚合日志的保存时间(可选) -->
  14. <name>yarn.log-aggregation.retain-seconds</name>
  15. <value>86400</value>
  16. </property>
  17. <property>
  18. <!-- 启用RM HA -->
  19. <name>yarn.resourcemanager.ha.enabled</name>
  20. <value>true</value>
  21. </property>
  22. <property>
  23. <!-- RM集群标识 -->
  24. <name>yarn.resourcemanager.cluster-id</name>
  25. <value>my-yarn-cluster</value>
  26. </property>
  27. <property>
  28. <!-- RM的逻辑ID列表 -->
  29. <name>yarn.resourcemanager.ha.rm-ids</name>
  30. <value>rm1,rm2</value>
  31. </property>
  32. <property>
  33. <!-- RM1的服务地址 -->
  34. <name>yarn.resourcemanager.hostname.rm1</name>
  35. <value>hadoop002</value>
  36. </property>
  37. <property>
  38. <!-- RM2的服务地址 -->
  39. <name>yarn.resourcemanager.hostname.rm2</name>
  40. <value>hadoop003</value>
  41. </property>
  42. <property>
  43. <!-- RM1 Web应用程序的地址 -->
  44. <name>yarn.resourcemanager.webapp.address.rm1</name>
  45. <value>hadoop002:8088</value>
  46. </property>
  47. <property>
  48. <!-- RM2 Web应用程序的地址 -->
  49. <name>yarn.resourcemanager.webapp.address.rm2</name>
  50. <value>hadoop003:8088</value>
  51. </property>
  52. <property>
  53. <!-- ZooKeeper集群的地址 -->
  54. <name>yarn.resourcemanager.zk-address</name>
  55. <value>hadoop001:2181,hadoop002:2181,hadoop003:2181</value>
  56. </property>
  57. <property>
  58. <!-- 启用自动恢复 -->
  59. <name>yarn.resourcemanager.recovery.enabled</name>
  60. <value>true</value>
  61. </property>
  62. <property>
  63. <!-- 用于进行持久化存储的类 -->
  64. <name>yarn.resourcemanager.store.class</name>
  65. <value>org.apache.hadoop.yarn.server.resourcemanager.recovery.ZKRMStateStore</value>
  66. </property>
  67. </configuration>

5. mapred-site.xml

  1. <configuration>
  2. <property>
  3. <!--指定mapreduce作业运行在yarn上-->
  4. <name>mapreduce.framework.name</name>
  5. <value>yarn</value>
  6. </property>
  7. </configuration>

5. slaves

配置所有从属节点的主机名或IP地址,每行一个。所有从属节点上的DataNode服务和NodeManager服务都会被启动。

  1. hadoop001
  2. hadoop002
  3. hadoop003

4.4 分发程序

将Hadoop安装包分发到其他两台服务器,分发后建议在这两台服务器上也配置一下Hadoop的环境变量。

  1. # 将安装包分发到hadoop002
  2. scp -r /usr/app/hadoop-2.6.0-cdh5.15.2/ hadoop002:/usr/app/
  3. # 将安装包分发到hadoop003
  4. scp -r /usr/app/hadoop-2.6.0-cdh5.15.2/ hadoop003:/usr/app/

五、启动集群

5.1 启动ZooKeeper

分别到三台服务器上启动ZooKeeper服务:

  1. zkServer.sh start

5.2 启动Journalnode

分别到三台服务器的的${HADOOP_HOME}/sbin目录下,启动journalnode进程:

  1. hadoop-daemon.sh start journalnode

5.3 初始化NameNode

hadop001上执行NameNode初始化命令:

  1. hdfs namenode -format

执行初始化命令后,需要将NameNode元数据目录的内容,复制到其他未格式化的NameNode上。元数据存储目录就是我们在hdfs-site.xml中使用dfs.namenode.name.dir属性指定的目录。这里我们需要将其复制到hadoop002上:

  1. scp -r /home/hadoop/namenode/data hadoop002:/home/hadoop/namenode/

5.4 初始化HA状态

在任意一台NameNode上使用以下命令来初始化ZooKeeper中的HA状态:

  1. hdfs zkfc -formatZK

5.5 启动HDFS

进入到hadoop001${HADOOP_HOME}/sbin目录下,启动HDFS。此时hadoop001hadoop002上的NameNode服务,和三台服务器上的DataNode服务都会被启动:

  1. start-dfs.sh

5.6 启动YARN

进入到hadoop002${HADOOP_HOME}/sbin目录下,启动YARN。此时hadoop002上的ResourceManager服务,和三台服务器上的NodeManager服务都会被启动:

  1. start-yarn.sh

需要注意的是,这个时候hadoop003上的ResourceManager服务通常是没有启动的,需要手动启动:

  1. yarn-daemon.sh start resourcemanager

六、查看集群

6.1 查看进程

成功启动后,每台服务器上的进程应该如下:

  1. [root@hadoop001 sbin]# jps
  2. 4512 DFSZKFailoverController
  3. 3714 JournalNode
  4. 4114 NameNode
  5. 3668 QuorumPeerMain
  6. 5012 DataNode
  7. 4639 NodeManager
  8. [root@hadoop002 sbin]# jps
  9. 4499 ResourceManager
  10. 4595 NodeManager
  11. 3465 QuorumPeerMain
  12. 3705 NameNode
  13. 3915 DFSZKFailoverController
  14. 5211 DataNode
  15. 3533 JournalNode
  16. [root@hadoop003 sbin]# jps
  17. 3491 JournalNode
  18. 3942 NodeManager
  19. 4102 ResourceManager
  20. 4201 DataNode
  21. 3435 QuorumPeerMain

6.2 查看Web UI

HDFS和YARN的端口号分别为500708080,界面应该如下:

此时hadoop001上的NameNode处于可用状态:

而hadoop002上的NameNode则处于备用状态:

hadoop002上的ResourceManager处于可用状态:

hadoop003上的ResourceManager则处于备用状态:

同时界面上也有Journal Manager的相关信息:

七、集群的二次启动

上面的集群初次启动涉及到一些必要初始化操作,所以过程略显繁琐。但是集群一旦搭建好后,想要再次启用它是比较方便的,步骤如下(首选需要确保ZooKeeper集群已经启动):

hadoop001启动 HDFS,此时会启动所有与 HDFS 高可用相关的服务,包括 NameNode,DataNode 和 JournalNode:

  1. start-dfs.sh

hadoop002启动YARN:

  1. start-yarn.sh

这个时候hadoop003上的ResourceManager服务通常还是没有启动的,需要手动启动:

  1. yarn-daemon.sh start resourcemanager

参考资料

以上搭建步骤主要参考自官方文档:

关于Hadoop高可用原理的详细分析,推荐阅读:

Hadoop NameNode 高可用 (High Availability) 实现解析

更多大数据系列文章可以参见个人 GitHub 开源项目: 程序员大数据入门指南

Hadoop 学习之路(八)—— 基于ZooKeeper搭建Hadoop高可用集群的更多相关文章

  1. Spark学习之路(七)—— 基于ZooKeeper搭建Spark高可用集群

    一.集群规划 这里搭建一个3节点的Spark集群,其中三台主机上均部署Worker服务.同时为了保证高可用,除了在hadoop001上部署主Master服务外,还在hadoop002和hadoop00 ...

  2. 基于 ZooKeeper 搭建 Spark 高可用集群

    一.集群规划 二.前置条件 三.Spark集群搭建         3.1 下载解压         3.2 配置环境变量         3.3 集群配置         3.4 安装包分发 四.启 ...

  3. Spark 系列(七)—— 基于 ZooKeeper 搭建 Spark 高可用集群

    一.集群规划 这里搭建一个 3 节点的 Spark 集群,其中三台主机上均部署 Worker 服务.同时为了保证高可用,除了在 hadoop001 上部署主 Master 服务外,还在 hadoop0 ...

  4. 入门大数据---基于Zookeeper搭建Spark高可用集群

    一.集群规划 这里搭建一个 3 节点的 Spark 集群,其中三台主机上均部署 Worker 服务.同时为了保证高可用,除了在 hadoop001 上部署主 Master 服务外,还在 hadoop0 ...

  5. Kafka 学习之路(二)—— 基于ZooKeeper搭建Kafka高可用集群

    一.Zookeeper集群搭建 为保证集群高可用,Zookeeper集群的节点数最好是奇数,最少有三个节点,所以这里搭建一个三个节点的集群. 1.1 下载 & 解压 下载对应版本Zookeep ...

  6. Kafka —— 基于 ZooKeeper 搭建 Kafka 高可用集群

    一.Zookeeper集群搭建 为保证集群高可用,Zookeeper集群的节点数最好是奇数,最少有三个节点,所以这里搭建一个三个节点的集群. 1.1 下载 & 解压 下载对应版本Zookeep ...

  7. Kafka 系列(二)—— 基于 ZooKeeper 搭建 Kafka 高可用集群

    一.Zookeeper集群搭建 为保证集群高可用,Zookeeper 集群的节点数最好是奇数,最少有三个节点,所以这里搭建一个三个节点的集群. 1.1 下载 & 解压 下载对应版本 Zooke ...

  8. 入门大数据---基于Zookeeper搭建Kafka高可用集群

    一.Zookeeper集群搭建 为保证集群高可用,Zookeeper 集群的节点数最好是奇数,最少有三个节点,所以这里搭建一个三个节点的集群. 1.1 下载 & 解压 下载对应版本 Zooke ...

  9. 基于keepalived搭建MySQL高可用集群

    MySQL的高可用方案一般有如下几种: keepalived+双主,MHA,MMM,Heartbeat+DRBD,PXC,Galera Cluster 比较常用的是keepalived+双主,MHA和 ...

  10. 基于Docker-compose搭建Redis高可用集群-哨兵模式(Redis-Sentinel)

    原文转载自「刘悦的技术博客」https://v3u.cn/a_id_110 我们知道,Redis的集群方案大致有三种:1)redis cluster集群方案:2)master/slave主从方案:3) ...

随机推荐

  1. Delphi 禁用x64位系统文件重定向功能

    在X64系统里面,一些特殊的目录和特殊的注册表键被分为2个独立的部分.对于文件系统来说,      %systemroot%\system32 目录被保留给64位文件使用,而32位文件会被重定向到%s ...

  2. Network Function Virtualization for a Network Device

    An apparatus for performing network function virtualization (NFV), comprising: a memory, a processor ...

  3. 自由WiFi软体,你也太简单了

    自由WiFi市场热点,自然不用多说.支付宝钱包计划实现全民免费WiFi,360.百度.金山.小米都在着手WiFi产品. 只是.这些免费WiFi或者实现WiFi的硬件或软件.都是争夺新的市场入口,推广产 ...

  4. 使用XCA(X Certificate and key management)可视化项目经理SSL 凭证(4)--凭借自身的凭证管理中心的定义(Certificate Authority)签名证书申请

    随着XCA(X Certificate and key management)可视化项目经理SSL 证书系列文章(2)和(3)中.我们学习了怎样用XCA(X Certificate and key m ...

  5. 半监督学习(semi-supervised learning)

    P(x)P(x,y)}⇒P(y|x) 自 P(x) 生成的无标签样本: 自 P(x,y) 生成的标记样本:

  6. C Shell中的变量数组

    今天刚刚在看一点C Shell的内容,发现一个挺好玩的东西!就是环境变量可以像数组那样来设置!具体设置语法如下: set variable=(element1 element2 ...) //注意元素 ...

  7. Vertica变化Local时间到GMT时间

    在Vertica的数据库的使用过程中碰到这么一种场景.程序从不同一时候区的集群中收集数据写入同一张表,然后我们须要把这些数据依照GMT时间来显示. 此时我们能够通过Vertica提供TIME ZONE ...

  8. WPF 遍历DataTemplate(获取所有控件)

    原文:WPF 遍历DataTemplate(获取所有控件) 情况1:在设定DataTemplate的Name,并且他是在前台表示时,获取DataTemplate里的指定控件. 方法: http://b ...

  9. [转载]Delphi常用类型及定义单元

    原文地址:Delphi常用类型及定义单元作者:沧海一声笑 Delphi常用类型及定义单元-总结  sndplaysound                mmsystem Type Unit Date ...

  10. Bootstrap 分页翻页

    @{    Layout = null;}<!DOCTYPE html><html><head>    <meta name="viewport&q ...