不多说,直接上干货!

     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. ActiveMQ 安全认证

    修改配置文件 位置: apache-activemq-5.9.0/conf/ vi activemq.xml 在<broker xmlns="http://activemq.apach ...

  2. spring cloud-服务注册

    正常的服务模块,注册到注册中心,让别的服务发现,调用服务 创建“服务提供方” 下面我们创建提供服务的客户端,并向服务注册中心注册自己. 假设我们有一个提供计算功能的微服务模块,我们实现一个RESTfu ...

  3. FZU1977 Pandora adventure —— 插头DP

    题目链接:https://vjudge.net/problem/FZU-1977  Problem 1977 Pandora adventure Accept: 597    Submit: 2199 ...

  4. 一步一步学Silverlight 2系列(16):数据与通信之JSON

    概述 Silverlight 2 Beta 1版本发布了,无论从Runtime还是Tools都给我们带来了很多的惊喜,如支持框架语言Visual Basic, Visual C#, IronRuby, ...

  5. Tomcat的一些问题

    首先设置好TOMCAT环境变量(如何设置?) 1.用Servlet发现每次改动Servlet都要重启Tomcat才能生效!!!搞死爹了!! 可以选择shutdown命令或者直接叉掉汤姆猫窗口!!!然后 ...

  6. Android 动态注册 亮屏、息屏广播

    /***************************************************************************** * Android 动态注册 亮屏.息屏广 ...

  7. [Selenium] WebDriver 操作 HTML5 中的 drag/drop

    以 jQuery UI 官方网站元素测试,地址:http://jqueryui.com/draggable/ 示例: package com.learningselenium.html5; impor ...

  8. [laravel]请求处理

    请求进入 public/index.php 文件. bootstrap/start.php 文件创建应用程序对象并检测环境. 内部的 framework/start.php 文件配置相关设置并加载服务 ...

  9. 虚拟机bridged, NAT and host-only网络区别

    In Linux, a network of each type is created when running vmware-config.pl. In Windows, they are auto ...

  10. Oracle 安装报错 [INS-06101] IP address of localhost could not be determined 解决方法输入日志标题

    安装Oracle 11gR2,报错:[INS-06101] IP address of localhost could not be determined 出现这种错误是因为主机名和/etc/host ...