|
1.6 编译及调试Hadoop源代码
+ \& ~7 F8 `8 H读者在阅读源代码过程中, 可能需要修改部分源代码或者使用调试工具以便跟踪某些变量值变化过程, 此时要用到Hadoop
5 c2 C& n4 j/ D4 \/ ?# ^8 C- M源代码编译和调试方法。 本节将介绍Hadoop在伪分布式模式下的编译和调试方法, 其中调试方法主要介绍使用Eclipse远程调试工7 O; b4 e `& F3 a7 q3 D
具和打印调试日志两种。/ c4 k: Q% U7 a
Hadoop天生支持Linux而对其他操作系统( 如Windows) 很不友好, 本书也鼓励读者直接在Linux平台下编译和调试Hadoop源代
j& w, [. M3 A# \/ s码, 因此, 本节介绍的内容全部在Linux环境下。
% V- \1 n5 X( C# c3 e1.6.1 编译Hadoop源代码
- ?, z, _# i c n9 k5 X( B4 h在Linux环境下编译源代码之前, 需进行以下准备工作:* F$ G$ k, B" `; ^7 \0 t
❑确保安装的Maven版本在3.0.2以上;
" h* {1 V! h. ]❑Protocol Buffers安装版本为2.5.0;
* |" k0 X9 J8 A A |, ^8 [2 ]$ K❑如果要启用findbugs, 则需确认已经安装了Findbugs;
# C" s) i' R7 d$ J1 \6 h. {: Y/ Q❑如果要编译native code, 则需确认安装了CMake 2.6或者更新版本;2 b1 T& R" }* W1 y3 ^2 ?4 v) X
❑第一次编译代码, 需确认可以连接互联网( Maven要从代码库中下载依赖包) 。
# K6 a3 x. b& l2 J" }. JMaven编译命令如表1-2所示。5 t+ B8 k, e7 x& m0 X1 N* v: {
表1-2 Maven编译命令
" T& K+ w3 C$ N& C+ `3 o( }; @如果仅编译生成JAR包而无须编译native code、 测试用例和生成文档, 可在Hadoop安装目录下并输入以下命令( 推荐使用该
: X u K0 h9 v- Q5 O9 B命令编译Hadoop源代码) :
9 J/ @9 t4 O3 T: Fmvn package -Pdist -DskipTests -Dtar
8 q; B6 V: C' H- k7 m5 O; ]如果编译JAR包、 native code并生成文档, 可使用以下命令:
! ~7 Q. d* E2 r3 n$ L% Vmvn package -Pdist,native,docs -DskipTests -Dtar5 R5 f9 H( \% j( k
每个子模块编译后生成的JAR包放到了与源代码目录平级的target目录中, 比如ResourceManager的源代码目录是:! P0 l( w/ Y3 G9 L" U# S
${YARN_HOME}/hadoop-yarn-project/hadoop-yarn/hadoop-yarn-
9 G4 K7 m9 A1 K( o' U- b' Iserver/hadoop-yarnserver-resourcemanager/src# w1 _, i% O$ b$ ]$ Y! {
它对应生成JAR包放在了以下目录中:8 i# ~( K2 s5 O3 N+ ^
${YARN_HOME}/hadoop-yarn-project/hadoop-yarn/hadoop-yarnserver/hadoop-yarnserver-resourcemanager/target, G$ _ d5 T/ O- C, `( [2 ^9 f
如果修改了某个模块的代码, 可编译后, 将对应的JAR包覆盖到${HADOOP_HOME}/share/hadoop目录中对应的JAR包上。8 ]5 [5 Q" r; k6 l2 n8 {4 n7 g
如果仅编译Hadoop的某一个子模块, 需将该模块依赖的JAR包作为它的第三方库引入。 一种简单的实现方式是在Hadoop安装* d- P' Y) v* P7 a' }' Z: o$ {! q
目录下输入以下命令编译所有源代码:! R3 f) z' N4 F$ B% [
mvn install -DskipTests- G! f( a* c. _; D* k2 Y/ m5 d
然后进入子模块目录, 编译生成对应的JAR包。
' h# G6 ]4 r# b* I: W* P1.6.2 调试Hadoop源代码
& k. ?4 A- U& b- b本节介绍两种调试Hadoop源代码的方法: 利用Eclipse远程调试工具和打印调试日志。 这两种方法均可以调试伪分布式工作模* J$ F% `4 a* V; C
式和完全分布式工作模式下的Hadoop。 本节主要介绍伪分布式工作模式下的Hadoop调试方法。 r. P9 b: \$ N& }
( 1) 利用Eclipse进行远程调试
# U$ A! Z# H( ]1 b% K/ m8 ~下面以调试ResourceManager为例, 介绍利用Eclipse远程调试的基本方法, 这可分两步进行。
( X x7 e* D9 G, h1 w) `9 [$ G( `步骤1 调试模式下启动Hadoop。+ X1 l" \+ a ?9 r1 o; R; r
在Hadoop安装目录下运行如下的Shell脚本:0 ]1 l4 m# H* \$ S
export YARN_NODEMANAGER_OPTS="-Xdebug -Xrunjdwp:transport=dt_socket,address=8788,: e! `1 A G. ~0 l1 y8 U) u
server=y,suspend=y"# O1 p/ y" V+ _" n/ j& l- X) @
sbin/start-all.sh( k; B7 t: ?9 h+ {) V
运行了脚本后会看到Shell命令行终端显示如下信息:
0 P) I# S7 c: ?: U8 i- q5 \) G% \- _Listening for transport dt_socket at address: 8788
! \; E% l/ G M$ `0 w& U此时表明ResourceManager处于监听状态, 直到收到debug确认信息。
( ?) s' a7 R. j步骤2 设置断点。' x9 b# N% [5 F6 ^& ]
在前面新建的Java工程"hadoop-2.0"中, 找到ResourceManager相关代码, 并在感兴趣的地方设置一些断点。) L: o! [# E) Y2 g
步骤3 在Eclipse中调试Hadoop程序。$ {, h3 \: B! U* R( t
在Eclipse的菜单栏中, 依次选择"Run"→"Debug Configurations"→"Remote Java Applications", 并按照要求填写远程调试器名称) w! C- p8 h1 H, z. @
( 自己定义一个即可) , ResourceManager所在host以及监听端口号等信息, 并选择Hadoop源代码工程, 便可进入调试模式。- z3 |3 G- @" S
调试过程中, ResourceManager输出的信息被存储到日志文件夹下的yarn-XXX-resourcemanager-localhost.log文件( XXX为当前用6 V& D9 p. e# S$ n4 r4 ]
户名) 中, 可通过以下命令查看调试过程中打印的日志:
# f& q H8 C$ u' @5 Rtail -f logs/yarn-XXX-resourcemanager-localhost.log
7 a \) y- p% t& f! v4 o/ d. w" t( 2) 打印Hadoop调试日志' i& z: n) B3 n/ i4 p5 B% n
Hadoop使用了 Apache log4j [5] 作为基本日志库, 该日志库将日志分别5个级别, 分别为DEBUG、 INFO、 WARN、 ERROR和. H, L1 k& W+ y7 {" Q% V
FATAL。 这5个级别是有顺序的, 即DEBUG <INFO <WARN <ERROR <FATAL, 分别用来指定日志信息的重要程度。 日志输出规# N/ L4 X" D. A! q( W$ R- i5 e
则为: 只输出级别不低于设定级别的日志信息, 比如若级别设定为INFO, 则INFO、 WARN、 ERROR和FATAL级别的日志信息都* O! Q9 t/ o: c: j2 h$ ]
会输出, 但级别比INFO低的DEBUG则不会输出。
; L- V1 J# y7 {' w7 ~在Hadoop源代码中, 大部分Java文件中存在调试日志( DEBUG级别日志) , 但默认情况下, 日志级别是INFO, 为了查看更
/ }0 Z) k2 [( ^. f6 T详细的运行状态, 可采用以下几种方法打开DEBUG日志。
# t3 @0 T+ G* p0 Q6 j( J方法1 使用Hadoop Shell命令。
& W0 s( m; F$ V! U可使用Hadoop脚本中的daemonlog命令查看和修改某个类的日志级别, 比如, 可通过以下命令查看NodeManager类的日志级9 L7 ~( p4 J6 Z1 k' P4 n% }% h
别:
, F' c+ a& Q: S2 B) E* x; Ibin/hadoop daemonlog -getlevel ${nodemanager-host}:8042 \$ Q; c/ K' H4 X) B9 _3 ? F. x
org.apache.hadoop.yarn.server.nodemanager.NodeManager
6 i! W$ _6 v/ _. E: U3 U可通过以下命令将NodeManager类的日志级别修改为DEBUG:
/ f8 ] e3 t& w1 a2 I+ ]bin/hadoop daemonlog -setlevel ${nodemanager-host}:8042 \8 |: L! g |; ^, I" X8 d
org.apache.hadoop.yarn.server.nodemanager.NodeManager DEBUG5 B- C. u% ~- h4 n: ^3 i' I0 L& X% v
其中, nodemanager-host为NodeManager服务所在的host, 8042是NodeManager的HTTP端口号。
% @: ^0 c" \' E# |5 P7 W, g$ w方法2 通过Web界面。
0 d& x6 ]& t; g7 T4 {+ o% Z+ ~用户可以通过Web界面查看和修改某个类的日志级别, 比如, 可通过以下URL修改NodeManager类的日志级别:* | ~, T% {, Z% h1 o5 F/ h
http://${nodemanager-host}:8042/logLevel2 I, X# E; ]8 [3 _
方法3 修改log4j.properties文件。 r+ l+ j M! ]
以上两种方式只能暂时修改日志级别, 当Hadoop重启后会被重置, 如果要永久性改变日志级别, 可在目标节点配置目录下$ e; D& K" P4 g$ ^5 {
的log4j.properties文件中添加以下配置选项:( x+ J0 T; X0 [4 @
log4j.logger.org.apache.hadoop.yarn.server.nodemanager.NodeManager=DEBUG
% Q+ S1 D8 k% { ]! C5 E2 t此外, 有时为了专门调试某个Java文件, 需要把该文件的相关日志输出到一个单独文件中, 可在log4j.properties中添加以下内
; o, j m, c5 |' z$ k/ |+ K- `容:+ P! _; Y6 F! I: [) e
# 定义输出方式为自定义的TTOUT
/ p C+ w9 m% |$ G. Mlog4j.logger. org.apache.hadoop.yarn.server.nodemanager.NodeManager=DEBUG,TTOUT& G1 U+ P7 _9 ~: A' N7 j2 F
# 设置TTOUT 的输出方式为输出到文件7 l5 F" }; A, {, F+ A% P4 Y$ \
log4j.appender.TTOUT =org.apache.log4j.FileAppender
: v1 n& m+ ^8 O2 m# 设置文件路径+ _- q+ T4 s6 B' ]# ~, X* q. h
log4j.appender.TTOUT.File=${hadoop.log.dir}/NodeManager.log- e1 v# s) t* G1 S& S5 Z
# 设置文件的布局
# o# P1 F( h7 ilog4j.appender.TTOUT.layout=org.apache.log4j.PatternLayout
, C* | l0 [) {- M( F% \# 设置文件的格式
7 O, ~: ^% p+ t& Tlog4j.appender.TTOUT.layout.ConversionPattern=%d{ISO8601} %p %c: %m%n
/ V+ _0 m, ]' M3 Y0 ~这些配置选项会把NodeManager.java中的DEBUG日志写到日志目录下的NodeManager.log文件中。
" J7 a" f* \! T( }/ P" B, C在阅读源代码的过程中, 为了跟踪某个变量值的变化, 读者可能需要自己添加一些DEBUG日志。 在Hadoop源代码中, 大部
* c9 H9 O, V' T r* T1 a分类会定义一个日志打印对象, 通过该对象可打印各个级别的日志。 比如, 在NodeManager中用以下代码定义对象LOG:
0 S/ G$ }! W5 ]2 b y* V" u, u, Spublic static final Log LOG = LogFactory.getLog(NodeManager.class);
5 C, E0 h' f* E用户可使用LOG对象打印调试日志。 比如, 可在NodeManager的main函数首行添加以下代码:6 u+ i, ^# f/ L
LOG.debug("Start to lauch NodeManager...");" G/ o- p8 \: d4 i4 k8 y
然后重新编译Hadoop源代码, 并将org.apache.hadoop.yarn.server.nodemanager.NodeManager的调试级别修改为DEBUG, 重新启6 J3 q9 o) o: v; o u
动Hadoop后便可以看到该调试信息。
) A1 ~7 [. E J8 L( ][5] Apache log4j网址: http://logging.apache.org/log4j/index.html。
; B6 Z( Z- L2 |9 e# Z% {' ]1.7 小结
5 q' C' \$ F* y( G0 W7 g搭建一个高效的源代码学习环境是深入学习Hadoop的良好开端, 本章主要内容正是帮助读者搭建一个这样的学习环境。 在
/ p8 Q0 l* ?! d" Z笔者看来, 一个高效的Hadoop学习环境至少应该包括源代码阅读环境、 Hadoop使用环境和源代码编译调试环境, 而本章正是围/ h8 E/ L# E4 a! v! ~
绕这三个环境的搭建方法组织的。
" P! `4 k) S+ L5 A! _本章介绍了Linux环境下搭建Hadoop源代码阅读环境的方法, 在此基础上, 进一步介绍了Hadoop的基本使用方法, 主要涉及5 t8 n. Z1 V+ C9 d V
Hadoop Shell和Eclipse插件两种工具的使用。 最后介绍了Hadoop源代码编译和调试方法, 其中, 调试方法主要介绍了使用Eclipse远& L7 n% G; {; j z
程调试和打印调试日志两种。
( Q3 u" z) M& I. U" ^
: Y& f; K$ f4 x( ]& ~' R, G: ?. H; ]
7 P0 M6 O* e; m; W# z: h/ A |
|