不多说,直接上干货!

     Hadoop的MR作业支持链式处理,类似在一个生产牛奶的流水线上,每一个阶段都有特定的任务要处理,比如提供牛奶盒,装入牛奶,封盒,打印出厂日期,等等,通过这样进一步的分工,从而提高了生产效率,那么在我们的Hadoop的MapReduce中也是如此,支持链式的处理方式,这些Mapper像Linux管道一样,前一个Mapper的输出结果直接重定向到下一个Mapper的输入,形成一个流水线,而这一点与Lucene和Solr中的Filter机制是非常类似的,Hadoop项目源自Lucene,自然也借鉴了一些Lucene中的处理方式。

举个例子,比如处理文本中的一些禁用词,或者敏感词,等等,Hadoop里的链式操作,支持的形式类似正则Map+ Rrduce Map*,代表的意思是全局只能有一个唯一的Reduce,但是在Reduce的前后是可以存在无限多个Mapper来进行一些预处理或者善后工作的。

注意:

  1. 本人目前使用的版本是1.2.1,因此ChainMapper使用的还是old api。

  2. 老的API之中,只支持 N-Mapper + 1-Reducer的模式。 Reducer不在链式任务最开始即可。

比如:

  Map1 -> Map2 -> Reducer -> Map3 -> Map4

  (不确定在新版的API之中是否支持 N-Reducer的模式。不过new api 确实要简单简洁很多)

  在编程的时候,我们可以借用源码提供给我们的程序!在此基础上进行修改和编写。

比如我的源码本地目录如下:(找我的本地ChainMapper和ChainReducer案例

  D:\SoftWare\hadoop-2.2.0-src\hadoop-mapreduce-project\hadoop-mapreduce-client\hadoop-mapreduce-client-core\src\main\java\org\apache\hadoop\mapreduce\lib\chain

任务介绍:

  这个任务需要两步完成:

  1. 对一篇文章进行WordCount

  2. 统计出现次数超过5词的单词

WordCount我们很熟悉,因为版本限制,先使用old api 实现一次:

Java代码

  1. package hadoop_in_action_exersice;
  2. import java.io.IOException;
  3. import java.util.Iterator;
  4. import java.util.StringTokenizer;
  5. import org.apache.hadoop.fs.FileSystem;
  6. import org.apache.hadoop.fs.Path;
  7. import org.apache.hadoop.io.IntWritable;
  8. import org.apache.hadoop.io.LongWritable;
  9. import org.apache.hadoop.io.Text;
  10. import org.apache.hadoop.mapred.FileInputFormat;
  11. import org.apache.hadoop.mapred.FileOutputFormat;
  12. import org.apache.hadoop.mapred.JobClient;
  13. import org.apache.hadoop.mapred.JobConf;
  14. import org.apache.hadoop.mapred.MapReduceBase;
  15. import org.apache.hadoop.mapred.Mapper;
  16. import org.apache.hadoop.mapred.OutputCollector;
  17. import org.apache.hadoop.mapred.Reducer;
  18. import org.apache.hadoop.mapred.Reporter;
  19. import org.apache.hadoop.mapred.TextInputFormat;
  20. import org.apache.hadoop.mapred.TextOutputFormat;
  21. public class ChainedJobs {
  22. public static class TokenizeMapper extends MapReduceBase implements Mapper<LongWritable, Text, Text, IntWritable> {
  23. private final static IntWritable one = new IntWritable(1);
  24. public static final int LOW_LIMIT = 5;
  25. @Override
  26. public void map(LongWritable key, Text value,
  27. OutputCollector<Text, IntWritable> output, Reporter reporter)
  28. throws IOException {
  29. String line = value.toString();
  30. StringTokenizer st = new StringTokenizer(line);
  31. while(st.hasMoreTokens())
  32. output.collect(new Text(st.nextToken()), one);
  33. }
  34. }
  35. public static class TokenizeReducer extends MapReduceBase implements Reducer<Text, IntWritable, Text, IntWritable> {
  36. @Override
  37. public void reduce(Text key, Iterator<IntWritable> values,
  38. OutputCollector<Text, IntWritable> output, Reporter reporter)
  39. throws IOException {
  40. int sum = 0;
  41. while(values.hasNext()) {
  42. sum += values.next().get();
  43. }
  44. output.collect(key, new IntWritable(sum));
  45. }
  46. }
  47. public static void main(String[] args) throws IOException {
  48. JobConf conf = new JobConf(ChainedJobs.class);
  49. conf.setJobName("wordcount");           //设置一个用户定义的job名称
  50. conf.setOutputKeyClass(Text.class);    //为job的输出数据设置Key类
  51. conf.setOutputValueClass(IntWritable.class);   //为job输出设置value类
  52. conf.setMapperClass(TokenizeMapper.class);         //为job设置Mapper类
  53. conf.setCombinerClass(TokenizeReducer.class);      //为job设置Combiner类
  54. conf.setReducerClass(TokenizeReducer.class);        //为job设置Reduce类
  55. conf.setInputFormat(TextInputFormat.class);    //为map-reduce任务设置InputFormat实现类
  56. conf.setOutputFormat(TextOutputFormat.class);  //为map-reduce任务设置OutputFormat实现类
  57. // Remove output folder before run job(s)
  58. FileSystem fs=FileSystem.get(conf);
  59. String outputPath = "/home/hadoop/DataSet/Hadoop/WordCount-OUTPUT";
  60. Path op=new Path(outputPath);
  61. if (fs.exists(op)) {
  62. fs.delete(op, true);
  63. System.out.println("存在此输出路径,已删除!!!");
  64. }
  65. FileInputFormat.setInputPaths(conf, new Path("/home/hadoop/DataSet/Hadoop/WordCount"));
  66. FileOutputFormat.setOutputPath(conf, new Path(outputPath));
  67. JobClient.runJob(conf);         //运行一个job
  68. }
  69. }

上面是独立的一个Job,完成第一步。为了能紧接着完成第二步,我们需要在原来的基础上进行修改。

为了方便理解,上面的输入的例子如下:

Java代码

  1. accessed    3
  2. accessible  4
  3. accomplish  1
  4. accounting  7
  5. accurately  1
  6. acquire 1
  7. across  1
  8. actual  1
  9. actually    1
  10. add 3
  11. added   2
  12. addition    1
  13. additional  4

old api 的实现方式并不支持 setup() / cleanup() 操作这一点非常不好,因此在有可能的情况下最好还是要迁移到Hadoop 2.X

新的API会方便简洁很多

下面是增加了一个Mapper 来过滤

Java代码

  1. public static class RangeFilterMapper extends MapReduceBase implements Mapper<Text, IntWritable, Text, IntWritable> {
  2. @Override
  3. public void map(Text key, IntWritable value,
  4. OutputCollector<Text, IntWritable> output, Reporter reporter)
  5. throws IOException {
  6. if(value.get() >= LOW_LIMIT) {
  7. output.collect(key, value);
  8. }
  9. }
  10. }

这个Mapper做的事情很简单,就是针对每个key,如果他的value > LOW_LIMIT 那么就输出

所以,目前为止,任务链如下:

TokenizerMapper -> TokenizeReducer -> RangeFilterMapper

所以我们的main函数改成下面的样子:

Java代码

  1. public static void main(String[] args) throws IOException {
  2. JobConf conf = new JobConf(ChainedJobs.class);
  3. conf.setJobName("wordcount");           //设置一个用户定义的job名称
  4. //        conf.setOutputKeyClass(Text.class);    //为job的输出数据设置Key类
  5. //        conf.setOutputValueClass(IntWritable.class);   //为job输出设置value类
  6. //        conf.setMapperClass(TokenizeMapper.class);         //为job设置Mapper类
  7. //        conf.setCombinerClass(TokenizeReducer.class);      //为job设置Combiner类
  8. //        conf.setReducerClass(TokenizeReducer.class);        //为job设置Reduce类
  9. //        conf.setInputFormat(TextInputFormat.class);    //为map-reduce任务设置InputFormat实现类
  10. //        conf.setOutputFormat(TextOutputFormat.class);  //为map-reduce任务设置OutputFormat实现类
  11. // Step1 : mapper forr word count
  12. JobConf wordCountMapper  = new JobConf(false);
  13. ChainMapper.addMapper(conf,
  14. TokenizeMapper.class,
  15. LongWritable.class,     // input key type
  16. Text.class,             // input value type
  17. Text.class,             // output key type
  18. IntWritable.class,      // output value type
  19. false,                  //byValue or byRefference 传值还是传引用
  20. wordCountMapper);
  21. // Step2: reducer for word count
  22. JobConf wordCountReducer  = new JobConf(false);
  23. ChainReducer.setReducer(conf,
  24. TokenizeReducer.class,
  25. Text.class,
  26. IntWritable.class,
  27. Text.class,
  28. IntWritable.class,
  29. false,
  30. wordCountReducer);
  31. // Step3: mapper used as filter
  32. JobConf rangeFilterMapper  = new JobConf(false);
  33. ChainReducer.addMapper(conf,
  34. RangeFilterMapper.class,
  35. Text.class,
  36. IntWritable.class,
  37. Text.class,
  38. IntWritable.class,
  39. false,
  40. rangeFilterMapper);
  41. // Remove output folder before run job(s)
  42. FileSystem fs=FileSystem.get(conf);
  43. String outputPath = "/home/hadoop/DataSet/Hadoop/WordCount-OUTPUT";
  44. Path op=new Path(outputPath);
  45. if (fs.exists(op)) {
  46. fs.delete(op, true);
  47. System.out.println("存在此输出路径,已删除!!!");
  48. }
  49. FileInputFormat.setInputPaths(conf, new Path("/home/hadoop/DataSet/Hadoop/WordCount"));
  50. FileOutputFormat.setOutputPath(conf, new Path(outputPath));
  51. JobClient.runJob(conf);         //运行一个job
  52. }

下面是运行结果的一部分:

Java代码

  1. a   40
  2. and 26
  3. are 12
  4. as  6
  5. be  7
  6. been    8
  7. but 5
  8. by  5
  9. can 12
  10. change  5
  11. data    5
  12. files   7
  13. for 28
  14. from    5
  15. has 7
  16. have    8
  17. if  6
  18. in  27
  19. is  16
  20. it  13
  21. more    8
  22. not 5
  23. of  23
  24. on  5
  25. outputs 5
  26. see 6
  27. so  11
  28. that    11
  29. the 54

可以看到,英文之中,如果NLP不去除停用词(a, the, for ...) 等,效果确实会被大大的影响。

Hadoop的ChainMapper和ChainReducer使用案例(链式处理)(四)的更多相关文章

  1. Hadoop工作流--ChainMapper/ChainReducer?(三)

    不多说,直接上干货! Hadoop的ChainMapper和ChainReducer使用案例(链式处理) 什么是ChainMapper/ChainReducer?

  2. 组合式+迭代式+链式 MapReduce

    1.迭代式mapreduce 一些复杂的任务难以用一次mapreduce处理完成,需要多次mapreduce才能完成任务,例如Pagrank,Kmeans算法都需要多次的迭代,关于mapreduce迭 ...

  3. MR案例:链式ChainMapper

    类似于Linux管道重定向机制,前一个Map的输出直接作为下一个Map的输入,形成一个流水线.设想这样一个场景:在Map阶段,数据经过mapper01和mapper02处理:在Reduce阶段,数据经 ...

  4. Hadoop基础-Map端链式编程之MapReduce统计TopN示例

    Hadoop基础-Map端链式编程之MapReduce统计TopN示例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.项目需求 对“temp.txt”中的数据进行分析,统计出各 ...

  5. Hadoop生态圈-Knox网关的应用案例

    Hadoop生态圈-Knox网关的应用案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Knox网关简介 据Knox官网所述(http://knox.apache.org/) ...

  6. Apache Hadoop 2.9.2 的归档案例剖析

    Apache Hadoop 2.9.2 的归档案例剖析 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.   能看到这篇文章说明你对NameNode 工作原理是有深入的理解啦!我们知道 ...

  7. Hadoop生态圈-CDH与HUE使用案例

    Hadoop生态圈-CDH与HUE使用案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.HUE的介绍 1>.HUE的由来 HUE全称是HadoopUser Experi ...

  8. hadoop一代集群运行代码案例

    hadoop一代集群运行代码案例 集群 一个 master,两个slave,IP分别是192.168.1.2.192.168.1.3.192.168.1.4               hadoop版 ...

  9. Hadoop基础-MapReduce的Partitioner用法案例

    Hadoop基础-MapReduce的Partitioner用法案例 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Partitioner关键代码剖析 1>.返回的分区号 ...

随机推荐

  1. 通过反射获取java nio Direct Memory 的最大值和已使用值

    (ps:jdk1.7及之后可通过MBean获取这两个值)

  2. HBase在滴滴出行的应用场景和最佳实践

    摘要: 主要介绍了HBase和Phoenix在滴滴内部的一些典型案例.文章已在CSDN极客头条和<程序员>杂志发表,应朋友邀请,分享到云栖社区,希望给大家带来启发和帮助. 背景 对接业务类 ...

  3. 利用QBuffer和QLinkedList做数据块存储队列

    Qt中QByteArray存储数据很方便,使用QBuffer存储大块数据更方便.QBuffer类包装了QByteArray类对象,实际存储还是使用了QByteArray,但QBuffer实现了QIOD ...

  4. css hack (ie6-ie9+)

    IE6 css hack: 1. *html Selector {} /* Selector 表示 css选择器 下同 */ 2. Selector { _property: value; } /* ...

  5. Kafka在Windows安装运行

    摘要:本文主要说明了如何在Windows安装运行Kafka 一.安装JDK 过程比较简单,这里不做说明. 最后打开cmd输入如下内容,表示安装成功 二.安装zooeleeper 下载安装包:http: ...

  6. web中使用svg失量图形及ie8以下浏览器的处理方式

    直接上代码: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <me ...

  7. 取clientdataset detal中的 更新数据, 将detal 转 数据库脚本sql

    转自永南博客,更改update 脚本只取变化字段,更改排除blob与数组字段,这两个类型会报错 function vartosql(value: Variant): wideString; var  ...

  8. contents属性

    CALayer 有一个属性叫做contents,这个属性的类型被定义为id,意味着它可以是任何类型的对象.在这种情况下,你可以给contents属性赋任何值,你的app仍然能够编译通过.但是,在实践中 ...

  9. html5--6-5 CSS选择器2

    html5--6-5 CSS选择器2 实例 学习要点 掌握常用的CSS选择器 了解不太常用的CSS选择器 什么是选择器 当我们定义一条样式时候,这条样式会作用于网页当中的某些元素,所谓选择器就是样式作 ...

  10. No tests found with test runner 'JUnit 3'

    报异常:No tests found with test runner 'JUnit 3' 解决方案: 主要因为你当前建的JUnit类是3的版本,将该类备份,重新创建一个类. 1.右键目录New--O ...