环境:CentOS 6.5, OPenJDK 1.7, Hadoop 2.2.0
本文主要参考官网的文档,Hadoop 2.2.0 Single Node Setup, Hadoop 2.2.0 Cluster Setup
一般我倾向于把需要启动daemon进程,对外提供服务的程序,简单的说,就是服务器类程序,安装在单独的用户下面。这样可以做到隔离,运维方面,安全性也提高了。
创建一个新的group,
$ sudo groupadd hadoop
创建一个新的用户,并加入group,
$ sudo useradd -g hadoop hadoop
给新用户设置密码,
$ sudo passwd hadoop
Hadoop能在单台机器上以伪分布式模式运行,即每个Hadoop模块运行在单独的java进程里。
先检查一下是能够无密码登录本机,
ssh localhost
如果提示输入密码,说明不能,按如下步骤设置。
$ ssh-keygen -t dsa -P '' -f ~/.ssh/id_dsa
$ cat ~/.ssh/id_dsa.pub >> ~/.ssh/authorized_keys
用浏览器下载或wget,
$ wget http://mirrors.hust.edu.cn/apache/hadoop/common/hadoop-2.2.0/hadoop-2.2.0.tar.gz
$ tar -zxf hadoop-2.2.0.tar.gz -C ~/local/opt
$ cd ~/local/opt/hadoop-2.2.0
$ vim ~/.bashrc
export HADOOP_PREFIX=$HOME/local/opt/hadoop-2.2.0
export HADOOP_COMMON_HOME=$HADOOP_PREFIX
export HADOOP_HDFS_HOME=$HADOOP_PREFIX
export HADOOP_MAPRED_HOME=$HADOOP_PREFIX
export HADOOP_YARN_HOME=$HADOOP_PREFIX
export HADOOP_CONF_DIR=$HADOOP_PREFIX/etc/hadoop
export PATH=$PATH:$HADOOP_PREFIX/bin:$HADOOP_PREFIX/sbin
配置文件的位置在 $HADOOP_PREIFIX/etc/hadoop
下面。
在这个文件中要告诉hadoop JDK 安装在了哪里
$ echo $JAVA_HOME
/usr/lib/jvm/java
$ vim conf/hadoop-env.sh
取消JAVA_HOME
那一行的注释,设置正确的JDK位置
export JAVA_HOME=/usr/lib/jvm/java
为了简单,我们仍采用Hadoop 1.x中的HDFS工作模式(不配置HDFS Federation)。
core-site.xml:
fs.defaultFS
hdfs://localhost
hdfs-site.xml:
dfs.datanode.data.dir
file:///home/hadoop/local/var/hadoop/hdfs/datanode
dfs.namenode.name.dir
file:///home/hadoop/local/var/hadoop/hdfs/namenode
dfs.namenode.checkpoint.dir
file:///home/hadoop/local/var/hadoop/hdfs/namesecondary
dfs.replication
1
Hadoop会自动创建目录。
yarn-site.xml,不用修改,保持为空。
Yarn不仅仅只支持MapReduce这种计算模式,还支持Spar, Tez, MPI等框架,因此,要显式地配置Yarn,使其支持mapreduce。在Hadoop 1.x中,只支持MapReduce,因此需要而且必须配置mapred-site.xml,到了Hadoop 2.x,MapReduce的配置是可选的。
在yarn-site.xml中添加:
yarn.nodemanager.aux-services
mapreduce_shuffle
解释:为了能够运行MapReduce程序,需要让各个NodeManager在启动时加载shuffle server,shuffle server实际上是Jetty/Netty Server,Reduce Task通过该server从各个NodeManager上远程拷贝Map Task产生的中间结果。上面增加的这个配置用于指定shuffle server。
将 mapred-site.xml.template 复制一份,重命名为mapred-site.xml。
mapred-site.xml:
mapreduce.framework.name
yarn
解释:用mapreduce.framework.name指定采用的框架名称,默认是将作业提交到MRv1的JobTracker端。
$ hdfs namenode -format
$ start-dfs.sh
$ start-yarn.sh
在Hadoop 2.x中,MapReduce Job不需要额外的daemon进程,在MapReduce Application Master启动的时候会自动启动JobTracker和TaskTracker进程,Job结束的时候会自动被关闭。
运行一个Hadoop自带的例子,名称为DistributedShell
,可以同时在多台机器上运行shell命令。
$ hadoop jar $HADOOP_PREFIX/share/hadoop/yarn/hadoop-yarn-applications-distributedshell-2.2.0.jar org.apache.hadoop.yarn.applications.distributedshell.Client --jar $HADOOP_PREFIX/share/hadoop/yarn/hadoop-yarn-applications-distributedshell-2.2.0.jar --shell_command date --num_containers 2
运行完成后,看看倒数第三行,有类似与application_1391783685869_0001
的字符串,这是application ID。查看该application的每个container的输出,执行下面的命令,
$ grep "" $HADOOP_PREFIX/logs/userlogs//**/stdout
$ cd $HADOOP_PREFIX
$ hdfs dfs -put ./etc/hadoop/ input
$ hadoop jar $HADOOP_PREFIX/share/hadoop/mapreduce/hadoop-mapreduce-examples-2.2.0.jar wordcount input output
$ hdfs dfs -lsr output
$ hdfs dfs -cat output/part-r-00000
结束后,关闭 Hadoop:
$ stop-dfs.sh
$ stop-yarn.sh
如果你已经有了三台机器,这一步可以省略。
如果没有,则可以用VMware Workstation 或 VirtualBox创建3台虚拟机。首先用vmware workstation 新建一台CentOS 6.5,装好操作系统,选择 Basic Server,安装JDK,参考我的另一篇博客,安装和配置CentOS服务器的详细步骤。安装好后然后用浅拷贝Create a linked clone
克隆出2台,这样有了3台虚拟机。启动3台机器,假设IP分别为192.168.1.131, 192.168.1.132, 192.168.1.133
, 131做为NameNode,JobTracker和SecondaryNameNode,身兼3个角色,这3个角色应该放到3台不同的机器上,这里为了简化,用一台机器来做3个角色;132和133为 slave。假设三台机器上的用户名是hadoop
,也可以用其他用户名,但必须三台机器都相同。
临时关闭防火墙
$ sudo service iptables stop
下次开机后,防火墙还是会启动。
永久关闭防火墙
$ sudo chkconfig iptables off
由于这几台虚拟机是开发机,不是生产环境,因此不必考虑安全性,可以永久关闭防火墙,还能给开发阶段带来很多便利。
如果集群中的每一台机器事先已经有了hostname,则这一步可以跳过。
这一步看起来貌似不必要,其实是必须的,否则最后运行wordcount等例子时,会出现“Too many fetch-failures”。因为HDFS用hostname而不是IP,来相互之间进行通信(见后面的注意1)。
在CentOS上修改hostname,包含两个步骤(假设将hostname1改为hostname2,参考这里,但不需要第一步):
/etc/sysconfig/network
內的 HOSTNAME 改成 yourhostnamehostname
命令,临时修改机器名, sudo hostname yourhostname
用exit
命令退出shell,再次登录,命令提示字符串就会变成[username@yourhostname ~]$
。
用上述方法,将131改名为master,132改名为slave01,133改名为slave02。
注意,对于有的Ubuntu/Debia 系统,会把本机的hostname解析成 127.0.1.1,例如:
127.0.0.1 localhost
127.0.1.1 master
将第二行改为(参考利用Cloudera实现Hadoop)
127.0.0.1 master
/etc/hosts
文件在所有机器的/etc/hosts
文件中,添加所有hostname对应的IP,一般在一台机器上设置好,然后scp到所有机器。例如
192.168.1.131 master
192.168.1.132 slave01
192.168.1.133 slave02
参考我的另一篇博客,SSH无密码登录的配置
将 hadoop-1.2.1-bin.tar.gz 上传到所有机器,然后解压。注意,所有机器的hadoop路径必须一致,因为master会登陆到slave上执行命令,master认为slave的hadoop路径与自己一样。
下面开始配置,配置好了后,把./etc/hadoop
目录scp到所有其他机器。
在第1节的基础上,增加下列修改。
在core-site.xml中,fs.defaultFS
要改为运行NameNode的那台机器的hostname,不再是localhost。
fs.defaultFS
hdfs://master
在yarn-site.xml中增加,
yarn.resourcemanager.hostname
master
在 etc/hadoop/slaves
中添加,
slave01
slave02
在core-site.xml里添加:
hadoop.tmp.dir
/home/hadoop/local/var/hadoop/tmp/hadoop-${user.name}
添加如下内容:
mapreduce.jobtracker.staging.root.dir
/user
这是为以后的多用户支持做准备。
在hadoop-env.sh中添加
export HADOOP_MAPRED_PID_DIR=/home/hadoop/local/var/hadoop/pids
在 mapred-env.sh中添加
export HADOOP_PID_DIR=/home/hadoop/local/var/hadoop/pids
我们这里有2台slave,就设置为2。在hdfs-site.xml里添加:
dfs.replication
2
$ cd $HADOOP_PREFIX/etc/hadoop
$ scp hadoop-env.sh mapred-env.sh core-site.xml hdfs-site.xml mapred-site.xml yarn-site.xml slaves hadoop@slave01:$HADOOP_PREFIX/etc/hadoop
$ scp hadoop-env.sh mapred-env.sh core-site.xml hdfs-site.xml mapred-site.xml yarn-site.xml slaves hadoop@slave02:$HADOOP_PREFIX/etc/hadoop
在所有机器上添加环境变量,与第1.3节相同。
在NameNode这个机器(在这里是master)上执行下列命令,
#只需一次,下次启动不再需要格式化,只需 start-dfs.sh
$ hdfs namenode -format
$ start-dfs.sh
在ResourceManager这台机器(在这里仍然是master)上执行,
$ start-yarn.sh
在Hadoop 2.x中,MapReduce Job不需要额外的daemon进程,在Job开始的时候,NodeManager会启动一个MapReduce Application Master(相当与一个精简的JobTracker),Job结束的时候自动被关闭。
用jps
查看java进程。
在master上,应该有三个进程,NameNode, SecondaryNameNode, ResourceManger;在每台slave上,应该有两个进程,DataNode, NodeManager。
可以用浏览器打开NameNode, ResourceManager和各个NodeManager的web界面,
还可以启动JobHistory Server,能够通过Web页面查看集群的历史Job,执行如下命令:
mr-jobhistory-daemon.sh start historyserver
默认使用19888端口,通过访问http://master:19888/查看历史信息。
终止JobHistory Server,执行如下命令:
mr-jobhistory-daemon.sh stop historyserver
将输入数据拷贝到HDFS中:
$ cd $HADOOP_PREFIX
$ hdfs dfs -put ./etc/hadoop input
这一步会报错,”No such file or directory”, 用hdfs dfs -ls /
查看,是空的,难怪了。我们需要手动建立”/user/hadoop”目录,
$ hdfs dfs -mkdir /user/hadoop
再上传文件,就可以了。
运行WordCount:
$ hadoop jar $HADOOP_PREFIX/share/hadoop/mapreduce/hadoop-mapreduce-examples-2.2.0.jar wordcount input output
查看结果:
$ hdfs dfs -lsr output
$ hdfs dfs -cat output/part-r-00000
如果能看到结果,说明这个例子运行成功。
在master上执行:
$ stop-yarn.sh
$ stop-hdfs.sh
本文已经尽可能的把步骤详细列出来了,但是我相信大部分人不会一次成功。这时候,查找错误就很重要了。查找错误最重要的手段是查看hadoop的日志,一般在logs目录下。把错误消息复制粘贴到google,搜索一下,慢慢找错误。