MapReduce 详细教程

本文详细介绍了MapReduce编程框架,包括其定义、优缺点、核心思想及进程。讲解了WordCount案例,Hadoop序列化机制,以及自定义bean对象实现序列化的步骤。此外,深入探讨了MapReduce框架的InputFormat数据输入、Shuffle机制、Partition分区、数据压缩策略和YARN资源调度器的工作原理。文章最后讨论了MapReduce在企业环境中的优化方法,包括数据倾斜问题和小文件处理策略。
摘要由CSDN通过智能技术生成

文章目录

1. MapReduce 概述

1.1 MapReduce 定义

  MapReduce 是一个分布式运算程序的编程框架,是用户开发 “基于 Hadoop 的数据分析应用” 的核心框架。
  MapReduce 核心功能是将用户编写的业务逻辑代码和自带默认组件整合成一个完整的分布式运算程序,并发运行在一个 Hadoop 集群上。

1.2 MapReduce 优缺点

  1. 优点

    MapReduce 易于编程

    它简单的实现一些接口,就可以完成一个分布式程序,这个分布式程序可以分布到大量廉价的 PC 机器运行。也就是说你写一个分布式程序,跟写一个简单的串行程序是一模一样的。 就是因为这个特点使得 MapReduce 编程变得非常流行。

    良好的扩展性

    当你的计算资源不能得到满足的时候,你可以通过简单的增加机器来扩展它的计算能力。

    高容错性

    MapReduce 设计的初衷就是使程序能够部署在廉价的 PC 机器上,这就要求它具有很高的容错性。比如其中一台机器挂了,它可以把上面的计算任务转移到另外一个节点上面上运行,不至于这个任务运行失败,而且这个过程不需要人工参与,而完全是由 Hadoop 内部完成的。

    适合 PB 级以上海量数据的离线处理

    离线处理,说明它适合离线处理而不适合在线处理。比如像毫秒级别的返回一个结果,MapReduce 很难做到。

  2. 缺点

    MapReduce 不擅长做实时计算、流式计算、DAG(有向图)计算。

    实时计算

    MapReduce 无法像 Mysql 一样,在毫秒或者秒级内返回结果。

    流式计算

    流式计算的输入数据是动态的,而 MapReduce 的输入数据集是静态的,不能动态变化。这是因为 MapReduce 自身的设计特点决定了数据源必须是静态的。

    DAG(有向图)计算

    多个应用程序存在依赖关系,后一个应用程序的输入为前一个的输出。在这种情况下,MapReduce 并不是不能做,而是使用后,每个 MapReduce 作业的输出结果都会写入到磁盘,会造成大量的磁盘 IO,导致性能非常的低下。

1.3 MapReduce 核心思想

在这里插入图片描述

1.4 MapReduce 进程

  一个完整的 MapReduce 程序在分布式运行时有三类实例进程:

  1. MrAppMaster: 负责整个程序的过程调度及状态协调。
  2. Map Task: 负责Ma阶段的整个数据处理流程。
  3. ReduceTask: 负责Reduce阶段的整个数据处理流程。

1.5 MapReduce 编程规范

  用户编写的程序分成三个部分:Mapper,Reducer,Driver(提交运行 mr 程序的客户端)

  1. Mapper 阶段

    (1)用户自定义的 Mapper 要继承自己的父类
    (2)Mapper 的输入数据是 KV 对的形式(KV 的类型可自定义)
    (3)Mapper 中的业务逻辑写在 map() 方法中
    (4)Mapper 的输出数据是 KV 对的形式(KV的类型可自定义)
    (5)map() 方法(maptask 进程)对每一个 <K,V> 调用一次

  2. Reducer 阶段

    (1)用户自定义的 Reducer 要继承自己的父类
    (2)Reducer 的输入数据类型对应 Mapper 的输出数据类型,也是 KV
    (3)Reducer 的业务逻辑写在 reduce() 方法中
    (4)Reducetask 进程对每一组相同 K 的 <K,V> 组调用一次 reduce() 方法

  3. Driver 阶段

    整个程序需要一个 Drvier 来进行提交,提交的是一个描述了各种必要信息的 job 对象

1.6 WordCount 案例实操

1.6.1 需求

  在给定的文本文件中统计输出每一个单词出现的总次数

  1. 输入数据 hello.txt
atguigu atguigu
ss ss
cls cls
jiao
banzhang
xue
hadoop
  1. 期望输出数据
atguigu	2
banzhang	1
cls	2
hadoop	1
jiao	1
ss	2
xue	1

1.6.2 需求分析

在这里插入图片描述

1.6.3 编写程序

  1. 创建一个 maven 项目 mr
  2. 导入以下依赖
<dependencies>
   <dependency>
       <groupId>junit</groupId>
       <artifactId>junit</artifactId>
       <version>RELEASE</version>
   </dependency>
   <dependency>
       <groupId>org.apache.logging.log4j</groupId>
       <artifactId>log4j-core</artifactId>
       <version>2.8.2</version>
   </dependency>
   <dependency>
       <groupId>org.apache.hadoop</groupId>
       <artifactId>hadoop-common</artifactId>
       <version>2.7.7</version>
   </dependency>
   <dependency>
       <groupId>org.apache.hadoop</groupId>
       <artifactId>hadoop-client</artifactId>
       <version>2.7.7</version>
   </dependency>
   <dependency>
       <groupId>org.apache.hadoop</groupId>
       <artifactId>hadoop-hdfs</artifactId>
       <version>2.7.7</version>
   </dependency>
</dependencies>
  1. 在项目的 src/main/resources 目录下,新建一个文件,命名为 “log4j.properties”,在文件中填入以下内容:
log4j.rootLogger=INFO, stdout
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d %p [%c] - %m%n
log4j.appender.logfile=org.apache.log4j.FileAppender
log4j.appender.logfile.File=target/spring.log
log4j.appender.logfile.layout=org.apache.log4j.PatternLayout
log4j.appender.logfile.layout.ConversionPattern=%d %p [%c] - %m%n
  1. 编写 Mapper 类
package mr.wordcount;

import java.io.IOException;

import org.apache.hadoop.io.IntWritable;
import org.apache.hadoop.io.LongWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Mapper;


public class WordCountMapper extends Mapper<LongWritable, Text, Text, IntWritable> {
   

    Text k = new Text();
    IntWritable v = new IntWritable(1);

    @Override
    protected void map(LongWritable key, Text value, Context context) throws IOException, InterruptedException {
   

        // 1 获取一行
        String line = value.toString();

        // 2 切割
        String[] words = line.split(" ");

        // 3 输出
        for (String word : words) {
   
            k.set(word);
            context.write(k, v);
        }
    }
}
  1. 编写 Reducer 类
package mr.wordcount;

import java.io.IOException;

import org.apache.hadoop.io.IntWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Reducer;

public class WordCountReducer extends Reducer<Text, IntWritable, Text, IntWritable> {
   

    int sum;
    IntWritable v = new IntWritable();

    @Override
    protected void reduce(Text key, Iterable<IntWritable> values, Context context) throws IOException, InterruptedException {
   

        // 1 累加求和
        sum = 0;
        for (IntWritable count : values) {
   
            sum += count.get();
        }

        // 2 输出
        v.set(sum);
        context.write(key, v);
    }
}
  1. 编写 Driver 类
package mr.wordcount;

import java.io.IOException;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.io.IntWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Job;
import org.apache.hadoop.mapreduce.lib.input.FileInputFormat;
import org.apache.hadoop.mapreduce.lib.output.FileOutputFormat;

public class WordCountDriver {
   

    public static void main(String[] args) throws IOException, ClassNotFoundException, InterruptedException {
   

        // 1 获取job对象
        Configuration configuration = new Configuration();
        Job job = Job.getInstance(configuration);

        // 2 设置jar加载路径
        job.setJarByClass(WordCountDriver.class);

        // 3 关联map和reduce类
        job.setMapperClass(WordCountMapper.class);
        job.setReducerClass(WordCountReducer.class);

        // 4 设置map输出阶段的KV类型
        job.setMapOutputKeyClass(Text.class);
        job.setMapOutputValueClass(IntWritable.class);

        // 5 设置最终输出KV类型
        job.setOutputKeyClass(Text.class);
        job.setOutputValueClass(IntWritable.class);

        // 6 设置输入和输出路径
        FileInputFormat.setInputPaths(job, new Path(args[0]));
        FileOutputFormat.setOutputPath(job, new Path(args[1]));

        // 7 提交
//        job.submit();
        boolean result = job.waitForCompletion(true);

        System.exit(result ? 0 : 1);
    }
}
  1. 本地测试

    输入程序的输入路径和输出路径
    在这里插入图片描述

    运行后查看结果
    在这里插入图片描述
    在这里插入图片描述

  2. 集群上测试

    ① 在 pom.xml 里面设置编译版本(默认的 1.5 版本,编译时 idea 会报错)

<properties>
	<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <maven.compiler.source>1.8</maven.compiler.source>
    <maven.compiler.target>1.8</maven.compiler.target>
</properties>

  ② 将程序打成 jar 包,重命名为 wc.jar 然后拷贝到 Hadoop 集群中

  ③ 执行 WordCount 程序

hadoop jar wc.jar mr.wordcount.WordCountDriver /user/wordCount/input /user/wordcount/output

  ④ 查看结果
  在这里插入图片描述

2. Hadoop 序列化

2.1 序列化概述

  1. 什么是序列化

    序列化就是把内存中的对象,转换成字节序列(或其他数据传输协议)以便于存储(持久化)和网络传输。
    反序列化就是将收到字节序列(或其他数据传输协议)或者是硬盘的持久化数据,转换成内存中的对象。

  2. 为什么要序列化

    一般来说,“活的”对象只生存在内存里,关机断电就没有了。而且“活的”对象只能由本地的进程使用,不能被发送到网络上的另外一台计算机。然而序列化可以存储“活的”对象,可以将“活的”对象发送到远程计算机。

  3. 为什么不用 Java 的序列化

    Java 的序列化是一个重量级序列化框架(Serializable),一个对象被序列化后,会附带很多额外的信息(各种校验信息,header,继承体系等),不便于在网络中高效传输。所以,hadoop 自己开发了一套序列化机制(Writable),精简、高效。

  4. Hadoop 序列化的特点

    ① 紧凑:高效实用存储空间。
    ② 快速:读写数据的额外开销小。
    ③ 可扩展:随着通信协议的升级而可升级。
    ④ 互操作:支持多语言的交互。

  5. 常用数据序列化类型
    在这里插入图片描述

2.2 自定义 bean 对象实现序列化接口

  在企业开发中往往常用的基本序列化类型不能满足所有需求,比如在 Hadoop 框架内部传递一个 bean 对象,那么该对象就需要实现序列化接口。
  具体实现 bean 对象序列化步骤如下 7 步。

  1. 实现 Writable 接口
  2. 反序列化时,需要反射调用空参构造函数,所以必须有空参构造
public FlowBean() {
   
	super();
}
  1. 重写序列化方法
@Override
public void write(DataOutput out) throws IOException {
   
	out.writeLong(upFlow);
	out.writeLong(downFlow);
	out.writeLong(sumFlow);
}
  1. 重写反序列化方法
@Override
public void readFields(DataInput in) throws IOException {
   
	upFlow = in.readLong();
	downFlow = in.readLong();
	sumFlow = in.readLong();
}
  1. 反序列化的顺序和序列化的顺序完全一致
  2. 要想把结果显示在文件中,需要重写 toString(),可用 ”\t” 分开,方便后续用。
  3. 如果需要将自定义的 bean 放在 key 中传输,则还需要实现 Comparable 接口,因为 MapReduce 框中的 Shuffle 过程要求对 key 必须能排序。

2.3 序列化案例实操

2.3.1 需求

  统计每一个手机号耗费的总上行流量、下行流量、总流量

  1. 输入数据

    phone_data.txt

1	13736230513	192.196.100.1	2481	24681	200
2	13846544121	192.196.100.2	264	0	200
3 	13956435636	192.196.100.3	132	1512	200
4 	13966251146	192.168.100.1	240	0	404
5 	18271575951	192.168.100.2	1527	2106	200
6 	18841884132	192.168.100.3	4116	1432	200
7 	13590439668	192.168.100.4	1116	954	200
8 	15910133277	192.168.100.5	3156	2936	200
9 	13729199489	192.168.100.6	240	0	200
10 	13630577991	192.168.100.7	6960	690	200
11 	15043685818	192.168.100.8	3659	3538	200
12 	15959002129	192.168.100.9	1938	180	500
13 	13560439638	192.168.100.10	918	4938	200
14 	13470253144	192.168.100.11	180	180	200
15 	13682846555	192.168.100.12	1938	2910	200
16 	13992314666	192.168.100.13	3008	3720	200
17 	13509468723	192.168.100.14	7335	110349	404
18 	18390173782	192.168.100.15	9531	2412	200
19 	13975057813	192.168.100.16	11058	48243	200
20 	13768778790	192.168.100.17	120	120	200
21 	13568436656	192.168.100.18	2481	24681	200
22 	13568436656	192.168.100.19	1116	954	200
  1. 输入数据格式
    在这里插入图片描述
  2. 期望输出数据格式
    在这里插入图片描述

2.3.2 需求分析

在这里插入图片描述

2.3.3 编写 MapReduce 程序

1.编写流量统计的 Bean 对象

package mr.flowsum;

import java.io.DataInput;
import java.io.DataOutput;
import java.io.IOException;

import org.apache.hadoop.io.Writable;

// 1 实现writable接口
public class FlowBean implements Writable {
   

    private long upFlow;
    private long downFlow;
    private long sumFlow;

    //2  反序列化时,需要反射调用空参构造函数,所以必须有
    public FlowBean() {
   
        super();
    }

    public FlowBean(long upFlow, long downFlow) {
   
        super();
        this.upFlow = upFlow;
        this.downFlow = downFlow;
        this.sumFlow = upFlow + downFlow;
    }

    //3  写序列化方法
    @Override
    public void write(DataOutput out) throws IOException {
   
        out.writeLong(upFlow);
        out.writeLong(downFlow);
        out.writeLong(sumFlow);
    }

    //4 反序列化方法
    //5 反序列化方法读顺序必须和写序列化方法的写顺序必须一致
    @Override
    public void readFields(DataInput in) throws IOException {
   
        this.upFlow = in.readLong();
        this.downFlow = in.readLong();
        this.sumFlow = in.readLong();
    }

    // 6 编写toString方法,方便后续打印到文本
    @Override
    public String toString() {
   
        return upFlow + "\t" + downFlow + "\t" + sumFlow;
    }

    public long getUpFlow() {
   
        return upFlow;
    }

    public void setUpFlow(long upFlow) {
   
        this.upFlow = upFlow;
    }

    public long getDownFlow() {
   
        return downFlow;
    }

    public void setDownFlow(long downFlow) {
   
        this.downFlow = downFlow;
    }

    public long getSumFlow() {
   
        return sumFlow;
    }

    public void setSumFlow(long sumFlow) {
   
        this.sumFlow = sumFlow;
    }

    public void set(long downFlow, long upFlow) {
   
        this.downFlow = downFlow;
        this.upFlow = upFlow;
        this.sumFlow = downFlow + upFlow;
    }
}
  1. 编写 Mapper 类
package mr.flowsum;

import java.io.IOException;

import org.apache.hadoop.io.LongWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Mapper;

public class FlowCountMapper extends Mapper<LongWritable, Text, Text, FlowBean> {
   

    FlowBean v = new FlowBean();
    Text k = new Text();

    @Override
    protected void map(LongWritable key, Text value, Context context) throws IOException, InterruptedException {
   

        // 1 获取一行
        // 1	13736230513	192.196.100.1	2481	24681	200
        String line = value.toString();

        // 2 切割字段
        String[] fields = line.split("\t");

        // 3 封装对象
        // 取出手机号码
        String phoneNum = fields[1];

        // 取出上行流量和下行流量
        long upFlow = Long.parseLong(fields[3]);
        long downFlow = Long.parseLong(fields[4]);

        k.set(phoneNum);
        v.set(downFlow, upFlow);

        // 4 写出
        context.write(k, v);
    }
}
  1. 编写 Reducer 类
package mr.flowsum;

import java.io.IOException;

import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Reducer;

public class FlowCountReducer extends Reducer<Text, FlowBean, Text, FlowBean> {
   

    @Override
    protected void reduce(Text key, Iterable<FlowBean> values, Context context) throws IOException, InterruptedException {
   

        long sum_upFlow = 0;
        long sum_downFlow = 0;

        // 1 遍历所用bean,将其中的上行流量,下行流量分别累加
        for (FlowBean flowBean : values) {
   
            sum_upFlow += flowBean.getUpFlow();
            sum_downFlow += flowBean.getDownFlow();
        }

        // 2 封装对象
        FlowBean v = new FlowBean(sum_upFlow, sum_downFlow);

        // 3 写出
        context.write(key, v);
    }
}
  1. 编写 Driver 驱动类
package mr.flowsum;

import java.io.IOException;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.Path;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Job;
import org.apache.hadoop.mapreduce.lib.input.FileInputFormat;
import org.apache.hadoop.mapreduce.lib.output.FileOutputFormat;

public class FlowSumDriver {
   

    public static void main(String[] args) throws IllegalArgumentException, IOException, ClassNotFoundException, InterruptedException {
   

        // 输入输出路径需要根据自己电脑上实际的输入输出路径设置
        args = new String[]{
   "f:/0/input/inputflow.txt", "f:/0/output/flowSum"};

        // 1 获取配置信息,或者job对象实例
        Configuration configuration = new Configuration();
        Job job = Job.getInstance(configuration);

        // 2 指定本程序的jar包所在的本地路径
        job.setJarByClass(FlowSumDriver.class);

        // 3 指定本业务job要使用的mapper/Reducer业务类
        job.setMapperClass(FlowCountMapper.class);
        job.setReducerClass(FlowCountReducer.class);

        // 4 指定mapper输出数据的kv类型
        job.setMapOutputKeyClass(Text.class);
        job.setMapOutputValueClass(FlowBean.class);

        // 5 指定最终输出的数据的kv类型
        job.setOutputKeyClass(Text.class);
        job.setOutputValueClass(FlowBean.class);

        // 6 指定job的输入原始文件所在目录
        FileInputFormat.setInputPaths(job, new Path(args[0]));
        FileOutputFormat.setOutputPath(job, new Path(args[1]));

        // 7 将job中配置的相关参数,以及job所用的java类所在的jar包, 提交给yarn去运行
        boolean result = job.waitForCompletion(true);
        System.exit(result ? 0 : 1);
    }
}

3. MapReduce 框架原理

3.1 InputFormat 数据输入

3.1.1 切片与 MapTask 并行度决定机制

  MapTask 的并行度决定 Map 阶段的任务处理并发度,进而影响到整个 Job 的处理速度。

  数据块: Block 是 HDFS 物理上把数据分成一块一块。
  数据切片: 数据切片只是在逻辑上对输入进行分片,并不会在磁盘上将其切分成片进行存储。

在这里插入图片描述

3.1.2 Job 提交流程源码解析

在这里插入图片描述

3.1.3 FileInputFormat 切片源码解析

  1. 程序先找到你数据存储的目录。

  2. 开始遍历处理(规划切片)目录下的每一个文件

  3. 遍历第一个文件 ss txt

    ① 获取文件大小 fs.sizeOf(ss.txt)
    ② 计算切片大小
      computeSplitSize(Math.max(minSize,Math.min(maxSize, blocksize)) ) = blocksize = 128M
    ③ 默认情况下,切片大小 = blocksize
    ④ 开始切,形成第1个切片: ss.txt——0: 128M;第2个切片 ss.txt——128:256M;第3个切片 ss .txt——256M: 300M
      (每次切片时,都要判断切完剩下的部分是否大于块的 1.1 倍,不大于 1.1 倍就划分一块切片)
    ⑤ 将切片信息写到一个切片规划文件中
    ⑥ 整个切片的核心过程在 getSplit0 方法中完成
    ⑦ InputSplit 只记录了切片的元数据信息,比如起始位置、长度以及所在的节点列表等。

  4. 提交切片规划文件到 YARN上,YARN 上的 MrAppMaster 就可以根据切片规划文件计算开启 Map Task 个数。

3.1.4 FileInputFormat 切片机制

  1. 切片机制

    ① 简单地按照文件的内容长度进行切片
    ② 切片大小,默认等于 Block 大小
    ③ 切片时不考虑数据集整体,而是逐个针对每一 个文件单独切片

  2. 案例分析
    在这里插入图片描述

  3. 切片大小参数配置

    ① 源码中计算切片大小的公式
    在这里插入图片描述
    ② 切片大小设置
    在这里插入图片描述
    ③ 获取切片信息 API
    在这里插入图片描述

3.1.5 CombineTextInputFormat 切片机制

  框架默认的 TextInputFormat 切片机制是对任务按文件规划切片,不管文件多小,都会是一个单独的切片,都会交给一个 MapTask,这样如果有大量小文件,就会产生大量的 MapTask,处理效率极其低下。

  1. 应用场景

    CombineTextInputFormat 用于小文件过多的场景,它可以将多个小文件从逻辑上规划到一个切片中,这样,多个小文件就可以交给一个 MapTask 处理。

  2. 虚拟存储切片最大值设置

CombineTextInputFormat.setMaxInputSplitSize(job, 4194304);// 4m

  注意:虚拟存储切片最大值设置最好根据实际的小文件大小情况来设置具体的值。

  1. 切片机制
    在这里插入图片描述

  2. 案例实操

    将输入的 4 个小文件合并成一个切片统一处理。

    ① 不做任何处理,运行之前 WordCount 案例程序,观察切片个数。
    ② 在 WordCountDriver 中增加如下代码,运行程序,并观察运行的切片个数

// 如果不设置InputFormat,它默认用的是TextInputFormat.class
job.setInputFormatClass(CombineTextInputFormat.class);

//虚拟存储切片最大值设置4m
CombineTextInputFormat.setMaxInputSplitSize(job, 4194304);

3.1.6 FileInputFormat 实现类

  FileInputFormat 常见的接口实现类包括:TextInputFormat、KeyValueTextInputFormat.、NLineInputFormat、CombineTextInputFormat、SequenceFileInputFormat 和自定义 InputFormat 等。

  1. TextInputFormat

    TextInputFormat 是默认的 FileInputFormat 实现类。按行读取每条记录。键是存储该行在整个文件中的起始字节偏移量,LongWritable 类型。 值是这行的内容,不包括任何行终止符(换行符和回车符),Text 类型。

    以下是一 个示例,比如,一个分片包含了如下 4 条文本记录。

Rich learning form
Intelligent learning engine
Learning more convenient
From the real demand for more close to the enterprise

  每条记录表示为以下键/值对:

(0, Rich learning fo rm)
(20, Inte lligent learning engine)
(49, Learning more convenient)
(75, From the real demand for more close to the enterprise)
  1. KeyValueTextInputFormat

    每一行均为一条记录,被分隔符分割为 key,value。可以通过在驱动类中设置
    conf.set(KeyValueLineRecordReader.KEY _VALUE_SEPERATOR,“\t"),来设定分隔符。 默认分隔符是tab (\t) 。

    以下是一个示例, 输入是一个包含 4 条记录的分片。其中 一> 表示一个(水平方向的)制表符。

line1 一>Rich learning form 
line2 一>Intelligent learning engine
line3 一>Learning more convenient
line4 一>From the real demand for more close to the enterprise

  每条记录表示为以下键/值对:此时的键是每行排在制表符之前的 Text 序列。

(line1, Rich learning form)
(line2, Intelligent learning engine)
(line3, Learning more convenient)
(line4, From the real demand for more close to the enterprise)
  1. NLineInputFormat

    如果使用 NLineInputFormat,代表每个 map 进程处理的 InputSplit 不再按 Block 块去划分,而是按
    NLineInputFormat 指定的行数 N 来划分。即输入文件的总行数/N = 切片数,如果不整除,切片数 = 商+1。

    以下是一个示例,仍然以上面的 4 行输入为例。

Rich learning form
Intelligent learning engine
Learning more convenient
From the real demand for more close to the enterprise

  例如,如果 N 是 2,则每个输入分片包含两行。开启 2 个MapTask。

(0,Rich learning form)
(20. Intelligent learning engine)

  另一个 mapper 则收到后两行:

(49,Learning more convenient)
(75, From the real demand for more close to the enterprise)

  这里的键和值与 TextInputFormat 生成的一样。

3.1.7 KeyValueTextInputFormat 使用案例

  1. 需求

    统计输入文件中每一行的第一个单词相同的行数。

    ① 输入数据

banzhang ni hao
xihuan hadoop banzhang
banzhang ni hao
xihuan hadoop banzhang

  ② 期望结果数据

banzhang	2
xihuan	2
  1. 需求分析
    在这里插入图片描述

  2. 代码实现

    ① 编写 Mapper 类

package mr.kv;

import java.io.IOException;

import org.apache.hadoop.io.IntWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Mapper;

public class KVTextMapper extends Mapper<Text, Text, Text, IntWritable> {
   

    // 1 设置value
    IntWritable v = new IntWritable(1);

    @Override
    protected void map(Text key, Text value, Context context) throws IOException, InterruptedException {
   

        // banzhang ni hao
        // 2 写出
        context.write(key, v);
    }
}

  ② 编写 Reducer 类

package mr.kv;

import java.io.IOException;

import org.apache.hadoop.io.IntWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.</
  • 10
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值