Hadoop(九)—— MapReduce

文章目录

  • 1. MapReduce介绍
    • 1.1 MapReduce定义
    • 1.2 MapReduce优缺点
      • 1.2.1 优点
      • 1.2.2 缺点
    • 1.3 MapReduce核心思想
    • 1.4 MapReduce进程
    • 1.5 MapReduce编程规范
  • 2. MapReduce框架原理
    • 2.1 MapReduce工作流程
    • 2.2 MapTask工作机制
      • 2.2.1 并行度决定机制
      • 2.2.2 MapTask工作机制
    • 2.3 Shuffle工作机制
      • 2.3.1 工作机制
      • 2.3.2 Partition分区
        • 2.3.2.1 默认Partition分区
        • 2.3.2.2 自定义Partition分区
      • 2.3.3 WritableComparable排序
      • 2.3.4 Combiner合并
    • 2.4 ReduceTask工作机制
      • 2.4.1 设置ReduceTask并行度(个数)
      • 2.4.3 ReduceTask工作机制
  • 3. InputFormat和OutputFormat
    • 3.1 InputFormat数据输入
      • 3.1.1 job提交流程源码解析
      • 3.1.2 InputFormat接口实现类
        • 3.1.2.1 FileInputFormat
        • 3.1.2.2 TextInputFormat
        • 3.1.2.3 CombineTextInputFormat
        • 3.1.2.4 KeyValueTextInputFormat
        • 3.1.2.5 NLineInputFormat
    • 3.2 OutputFormat数据输出
  • 4. Join多种应用
    • 4.1 Map join(Distributedcache分布式缓存)
    • 4.2 Reduce join
  • 5. 数据清洗(ETL)
  • 6. 计数器应用
  • 7. MapReduce开发总结

1. MapReduce介绍

1.1 MapReduce定义

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

1.2 MapReduce优缺点

1.2.1 优点

  • 易于编程

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

  • 良好的扩展性

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

  • 高容错性

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

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

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

1.2.2 缺点

  • 不擅长做实时计算

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

  • 不擅长做流式计算

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

  • 不擅长做DAG(有向无环图)计算

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

1.3 MapReduce核心思想

Hadoop(九)—— MapReduce_第1张图片
Hadoop(九)—— MapReduce_第2张图片

  1. 分布式的运算程序往往需要分成至少2个阶段。
  2. 第一个阶段的MapTask并发实例,完全并行运行,互不相干。
  3. 第二个阶段的ReduceTask并发实例互不相干,但是他们的数据依赖于上一个阶段的所有MapTask并发实例的输出。
  4. MapReduce编程模型只能包含一个map阶段和一个reduce阶段,如果用户的业务逻辑非常复杂,那就只能多个MapReduce程序,串行运行。

1.4 MapReduce进程

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

  1. MRAppMaster:负责整个程序的过程调度及状态协调。[MR程序运行的时候会开启(JPS查看有MRAppMaster),程序运行结束的时候又会注销(JPS发现没MRAppMaster了)。]
  2. MapTask:负责map阶段的整个数据处理流程。
  3. ReduceTask:负责reduce阶段的整个数据处理流程。

1.5 MapReduce编程规范

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

Mapper阶段

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

Reducer阶段

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

Driver阶段

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

2. MapReduce框架原理

2.1 MapReduce工作流程

Hadoop(九)—— MapReduce_第3张图片Hadoop(九)—— MapReduce_第4张图片
上面的流程是整个MapReduce最全工作流程,但是shuffle过程只是从第7步开始到第15步结束,具体shuffle过程详解,如下:

  1. MapTask收集我们的map()方法输出的KV对,放到内存缓冲区中。
  2. 从内存缓冲区不断溢出本地磁盘文件,可能会溢出多个文件。
  3. 多个溢出文件会被合并成大的溢出文件。
  4. 在溢出过程中,及合并的过程中,都要调用partitioner进行分区和针对key进行排序。
  5. ReduceTask根据自己的分区号,去各个MapTask机器上取相应的结果分区数据。
  6. ReduceTask会取到同一个分区的来自不同MapTask的结果文件,ReduceTask会将这些文件再进行合并(归并排序)。
  7. 合并成大文件后,shuffle的过程也就结束了,后面进入ReduceTask的逻辑运算过程(从文件中取出一个一个的键值对group,调用用户自定义的reduce()方法)。

注:

  • Shuffle中的缓冲区大小会影响到MapReduce程序的执行效率,原则上说,缓冲区越大,磁盘IO的次数越少,执行速度就越快。缓冲区的大小可以通过参数调整,参数:io.sort.mb 默认100M。
  • 切片,默认一个块就是一个片。一个Map任务对应一个片。200M的话对应两个MapTask。切片对map的个数有关,跟reduce个数无关,分区数和reduce数对应。
  • 达到环形缓冲区的80%后,就开始溢写到磁盘,分区,排序(默认按字典排序),合并(可选,作用是预合并,减少reduce的压力)这三个过程都是在溢写过程中进行的。
  • 这里分成了两个区:每个区有3个溢写文件(200/80=3),每个溢写文件80M。

2.2 MapTask工作机制

2.2.1 并行度决定机制

MapTask的并行度决定map阶段的任务处理并发度,进而影响到整个job的处理速度。一个job的map阶段MapTask并行度(个数),由客户端提交job时的切片个数决定。 默认切片大小=blocksize,但记着1.1倍的事。

Hadoop(九)—— MapReduce_第5张图片

2.2.2 MapTask工作机制

Hadoop(九)—— MapReduce_第6张图片

  1. Read阶段:MapTask通过用户编写的RecordReader,从输入InputSplit中解析出一个个key/value。

  2. Map阶段:该节点主要是将解析出的key/value交给用户编写map()函数处理,并产生一系列新的key/value。

  3. Collect收集阶段:在用户编写map()函数中,当数据处理完成后,一般会调用OutputCollector.collect()输出结果。在该函数内部,它会将生成的key/value分区(调用Partitioner),并写入一个环形内存缓冲区中。

  4. Spill阶段:即“溢写”,当环形缓冲区满后,MapReduce会将数据写到本地磁盘上,生成一个临时文件。需要注意的是,将数据写入本地磁盘之前,先要对数据进行一次本地排序,并在必要时对数据进行合并、压缩等操作。

    溢写阶段详情:
    步骤1:利用快速排序算法对缓存区内的数据进行排序,排序方式是,先按照分区编号partition进行排序,然后按照key进行排序。这样,经过排序后,数据以分区为单位聚集在一起,且同一分区内所有数据按照key有序。
    步骤2:按照分区编号由小到大依次将每个分区中的数据写入任务工作目录下的临时文件output/spillN.out(N表示当前溢写次数)中。如果用户设置了Combiner,则写入文件之前,对每个分区中的数据进行一次聚集操作。
    步骤3:将分区数据的元信息写到内存索引数据结构SpillRecord中,其中每个分区的元信息包括在临时文件中的偏移量、压缩前数据大小和压缩后数据大小。如果当前内存索引大小超过1MB,则将内存索引写到文件output/spillN.out.index中。

  5. Combine阶段:当所有数据处理完成后,MapTask对所有临时文件进行一次合并,合并成一个大文件,并保存到文件output/file.out中,以确保每个MapTask最终只生成一个数据文件(避免同时打开大量文件和同时读取大量小文件产生的随机读取带来的开销)。同时生成相应的索引文件output/file.out.index。
    在进行文件合并过程中,MapTask以分区为单位进行合并。对于某个分区,它将采用多轮递归合并的方式。每轮合并io.sort.factor(默认100)个文件,并将产生的文件重新加入待合并列表中,对文件排序后,重复以上过程,直到最终得到一个大文件。

2.3 Shuffle工作机制

2.3.1 工作机制

MapReduce确保每个Reducer的输入都是按键排序的。系统执行排序的过程(即将map输出作为输入传给reducer)称为shuffle。

Hadoop(九)—— MapReduce_第7张图片
Hadoop(九)—— MapReduce_第8张图片
图解:

  • 进环形缓冲区的有数据+索引(元信息)

    元信息:记录临时文件偏移量,压缩大小等等。

    如果索引 > 1M,会写入到磁盘。

    索引和数据命名是有区分的。

  • 一个分组对应一个reduce方法;

    一个分区对应一个reduce任务。

2.3.2 Partition分区

2.3.2.1 默认Partition分区

public class HashPartitioner<K, V> extends Partitioner<K, V> {
  public int getPartition(K key, V value, int numReduceTasks) {
    // 默认分区是根据key的hashCode对reduceTasks个数取模得到的。用户没法控制哪个key存储到哪个分区。
    // &是为了不出现负数,也可以用 hashcode()%num + num来代替,但是位运算性能比较高。
    return (key.hashCode() & Integer.MAX_VALUE) % numReduceTasks;
  }
}

2.3.2.2 自定义Partition分区

自定义类继承Partitioner,重写getPartition()方法

package com.during.bigdata.Partition;

import org.apache.hadoop.io.NullWritable;
import org.apache.hadoop.io.Text;
import org.apache.hadoop.mapreduce.Partitioner;

import java.util.Arrays;

public class PhonePartition extends Partitioner<Text, NullWritable> {
    //移动号码前三位
    public static final String[] YD = {
            "134","135","136",
            "137","138","139",
            "150","151","152",
            "157","158","159",
            "188","187","182",
            "183","184","178",
            "147","172","198"
    };
    //电信号码前三位
    public static final String[] DX = {
            "133","149","153",
            "173","177","180",
            "181","189","199"};
    //联通号码前三位
    public static final String[] LT = {
            "130","131","132",
            "145","155","156",
            "166","171","175",
            "176","185","186","166"
    };

    public int getPartition(Text text, NullWritable nullWritable, int i) {
        String phone = text.toString().substring(0,3);
        //判断手机号的前三位属于哪个运营商
        if (Arrays.asList(YD).contains(phone)){
            return 1;
        }else if (Arrays.asList(DX).contains(phone)){
            return 2;
        }else if (Arrays.asList(LT).contains(phone)){
            return 3;
        } else {
          	return 0;
        }
    }
}

在Driver中设置

// 设置自定义Partitioner
job.setPartitionerClass(PhonePartition.class);
// 根据自定义Partitioner的逻辑设置相应数量的reduce task
job.setNumReduceTasks(4);

注:

  • 如果reduceTask的数量 > getPartition的结果数,则会多产生几个空的输出文件part-r-000xx;

  • 如果1 < reduceTask的数量 < getPartition的结果数,则有一部分分区数据无处安放,会报IOException;

  • 如果reduceTask的数量 = 1,则不管MapTask端输出多少个分区文件,最终结果都交给这一个ReduceTask,最终也就只会产生一个结果文件 part-r-00000;

  • 如果不设置,或者设置0,说明只有map,没有reduce。

    例如:假设自定义分区数为5,则
    job.setNumReduceTasks(1); 会正常运行,只不过会产生一个输出文件
    job.setNumReduceTasks(2); 会报错
    job.setNumReduceTasks(6); 大于5,程序会正常运行,会产生空文件

2.3.3 WritableComparable排序

排序是MapReduce框架中最重要的操作之一。MapTask和ReduceTask均会对数据(按照key)进行排序。该操作属于Hadoop的默认行为。任何应用程序中的数据均会被排序,而不管逻辑上是否需要。默认排序是按照字典顺序排序,且实现该排序的方法是快速排序
对于MapTask,它会将处理的结果暂时放到一个缓冲区中,当缓冲区使用率达到一定阈值后,再对缓冲区中的数据进行一次排序,并将这些有序数据写到磁盘上,而当数据处理完毕后,它会对磁盘上所有文件进行一次合并,以将这些文件合并成一个大的有序文件。
对于ReduceTask,它从每个MapTask上远程拷贝相应的数据文件,如果文件大小超过一定阈值,则放到磁盘上,否则放到内存中。如果磁盘上文件数目达到一定阈值,则进行一次合并以生成一个更大文件;如果内存中文件大小或者数目超过一定阈值,则进行一次合并后将数据写到磁盘上。当所有数据拷贝完毕后,ReduceTask统一对内存和磁盘上的所有数据进行一次合并。
每个阶段的默认排序

排序的分类:

  • 部分排序
    MapReduce根据输入记录的键对数据集排序。保证输出的每个文件内部排序。

  • 全排序
    最简单的方法是使用一个分区。但该方法在处理大型文件时效率极低,因为一台机器必须处理所有输出文件,从而完全丧失了MapReduce所提供的并行架构。
    替代方案:首先创建一系列排好序的文件;其次,串联这些文件;最后,生成一个全局排序的文件。主要思路是使用一个分区来描述输出的全局排序。例如:可以为上述文件创建3个分区,在第一分区中,记录的单词首字母a-g,第二分区记录单词首字母h-n, 第三分区记录单词首字母o-z。

  • 辅助排序(GroupingComparator分组)
    MapReduce框架在记录到达Reducer之前按键对记录排序,但键所对应的值并没有被排序。甚至在不同的执行轮次中,这些值的排序也不固定,因为它们来自不同的map任务且这些map任务在不同轮次中完成时间各不相同。一般来说,大多数MapReduce程序会避免让reduce函数依赖于值的排序。但是,有时也需要通过特定的方法对键进行排序和分组等以实现对值的排序。

    package com.during.bigdata.groupSort;
    
    import org.apache.hadoop.io.WritableComparable;
    import org.apache.hadoop.io.WritableComparator;
    
    public class Group  extends WritableComparator {
        //mapper类中的默认分组,Key
        public Group(){
            super(FlowBean.class,true);
        }
        @Override
        public int compare(WritableComparable a, WritableComparable b) {
            FlowBean a1 = (FlowBean) a;
            FlowBean b1 = (FlowBean) b;
            //按照FlowBean里面id重新分组
            if(a1.getId()>b1.getId()){
                return 1;
            }else if (a1.getId()<b1.getId()){
                return -1;
            }else {
                return 0;
            }
        }
    }
    

    Driver中

    //自定义分组,辅助排序分组的key
    job.setGroupingComparatorClass(Group.class);
    
  • 二次排序
    在自定义排序过程中,如果compareTo中的判断条件为两个即为二次排序。

    package com.during.bigdata.writablesort;
    
    import lombok.Getter;
    import lombok.Setter;
    import org.apache.hadoop.io.WritableComparable;
    
    import java.io.DataInput;
    import java.io.DataOutput;
    import java.io.IOException;
    @Getter
    @Setter
    public class FlowBean implements WritableComparable<FlowBean>{
        private long id;
        private double price;
      
        //二次排序
        public int compareTo(FlowBean o) {
            //按照订单id升序
            if (this.id>o.id){
                return 1;
            }else if (this.id<o.id){
                return -1;
            }else {
                //价格倒序
                return this.price>o.price? -1:1;
            }
        }
    }
    
  • 自定义排序WritableComparable
    bean对象实现WritableComparable接口重写compareTo方法,就可以实现排序

    package com.during.bigdata.writablecomparable;
    
    import lombok.Getter;
    import lombok.Setter;
    import org.apache.hadoop.io.WritableComparable;
    
    import java.io.DataInput;
    import java.io.DataOutput;
    import java.io.IOException;
    @Getter
    @Setter
    public class FlowBean implements WritableComparable<FlowBean> {
        private long upflow;
        private long downflow;
        private long sumflow;
      
        //自定义比较器,倒序排序
        public int compareTo(FlowBean o) {
            return this.sumFlow > o.sumflow() ? -1 : 1;
        }
    }
    

2.3.4 Combiner合并

方式一:对多个小文件使用CombineFileInputFormat

方式二:自定义Combiner

自定义一个Combiner继承Reducer,重写Reduce方法

public class WordCountCombiner extends Reducer<Text, IntWritable, Text, IntWritable>{
	@Override
	protected void reduce(Text key, Iterable<IntWritable> values,
			Context context) throws IOException, InterruptedException {
        // 1 汇总操作
				int count = 0;
        for(IntWritable v :values){
          count += v.get();
        }
        // 2 写出
				context.write(key, new IntWritable(count));
	}
}

Driver中设置:

job.setCombinerClass(WordcountCombiner.class);

2.4 ReduceTask工作机制

2.4.1 设置ReduceTask并行度(个数)

ReduceTask的并行度同样影响整个job的执行并发度和执行效率,但与MapTask的并发数由切片数决定不同,ReduceTask数量的决定是可以直接手动设置:

//默认值是1,手动设置为4
job.setNumReduceTasks(4);
  • ReduceTask = 0,表示没有reduce阶段,输出文件个数和map个数一致。
  • ReduceTask默认值就是1,所以输出文件个数为一个。
  • 如果数据分布不均匀,就有可能在reduce阶段产生数据倾斜。
  • ReduceTask数量并不是任意设置,还要考虑业务逻辑需求,有些情况下,需要计算全局汇总结果,就只能有1个ReduceTask。
  • 具体多少个ReduceTask,需要根据集群性能而定。
  • 如果分区数不是1,但是ReduceTask为1,是否执行分区过程。答案是:不执行分区过程。因为在MapTask的源码中,执行分区的前提是先判断reduceNum个数是否大于1。不大于1肯定不执行。

2.4.3 ReduceTask工作机制

Hadoop(九)—— MapReduce_第9张图片

  1. Copy阶段:ReduceTask从各个MapTask上远程拷贝一片数据,并针对某一片数据,如果其大小超过一定阈值,则写到磁盘上,否则直接放到内存中。
  2. Merge阶段:在远程拷贝数据的同时,ReduceTask启动了两个后台线程对内存和磁盘上的文件进行合并,以防止内存使用过多或磁盘上文件过多。
  3. Sort阶段:按照MapReduce语义,用户编写reduce()函数输入数据是按key进行聚集的一组数据。为了将key相同的数据聚在一起,Hadoop采用了基于排序的策略。由于各个MapTask已经实现对自己的处理结果进行了局部排序,因此,ReduceTask只需对所有数据进行一次归并排序即可。
  4. Reduce阶段:reduce()函数将计算结果写到HDFS上。

3. InputFormat和OutputFormat

3.1 InputFormat数据输入

3.1.1 job提交流程源码解析

waitForCompletion()
	submit();
	// 1建立连接
	connect();	
	// 创建提交job的代理
	new Cluster(getConfiguration());
	// 判断是本地yarn还是远程
  initialize(jobTrackAddr, conf); 
	// 2 提交job
	submitter.submitJobInternal(Job.this, cluster)
	// 1)创建给集群提交数据的Stag路径
	Path jobStagingArea = JobSubmissionFiles.getStagingDir(cluster, conf);
	// 2)获取jobid ,并创建job路径
	JobID jobId = submitClient.getNewJobID();
	// 3)拷贝jar包到集群
	copyAndConfigureFiles(job, submitJobDir);	
	rUploader.uploadFiles(job, jobSubmitDir);
	// 4)计算切片,生成切片规划文件
	writeSplits(job, submitJobDir);
	maps = writeNewSplits(job, jobSubmitDir);
	input.getSplits(job);
	// 5)向Stag路径写xml配置文件
	writeConf(conf, submitJobFile);
	conf.writeXml(out);
	// 6)提交job,返回提交状态
	status = submitClient.submitJob(jobId, submitJobDir.toString(), job.getCredentials());

3.1.2 InputFormat接口实现类

  • InputFormat

    • FileInputFormat

      • CombineFileInputFormat

      • FixedLengthInputFormat

      • SequeueFileInputFormat

      • KeyValueTextInputFormat

      • NLineInputFormat

      • TextInputFormat

  • 自定义InputFormat

    (1)自定义一个类继承FileInputFormat。

    (2)改写RecordReader,实现一次读取一个完整文件封装为KV。

    (3)在输出时使用SequenceFileOutPutFormat输出合并文件。

3.1.2.1 FileInputFormat

  • FileInputFormat中默认的切片机制

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

例如,待处理数据有两个文件:
file1.txt 320M
file2.txt 10M

经过FileInputFormat的切片机制运算后,形成的切片信息如下:
file1.txt.split1-- 0~128
file1.txt.split2-- 128~256
file1.txt.split3-- 256~320
file2.txt.split1-- 0~10M

  • FileInputFormat切片大小的参数配置
    通过分析源码,在FileInputFormat的280行中,计算切片大小的逻辑:

    Math.max(minSize, Math.min(maxSize, blockSize));
    切片主要由这几个值来运算决定
    mapreduce.input.fileinputformat.split.minsize=1,默认值为1
    mapreduce.input.fileinputformat.split.maxsize=Long.MAXValue,默认值Long.MAXValue
    因此,默认情况下,切片大小=blocksize。
    maxsize(切片最大值):
    参数如果调得比blocksize小,则会让切片变小,而且就等于配置的这个参数的值。
    minsize(切片最小值):
    参数调的比blockSize大,则可以让切片变得比blocksize还大。

  • 获取切片信息API

    // 根据文件类型获取切片信息
    FileSplit inputSplit = (FileSplit) context.getInputSplit();
    // 获取切片的文件名称
    String name = inputSplit.getPath().getName();
    
  • 源码解析(input.getSplits(job))

  1. 找到数据存储的目录。

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

  3. 遍历第一个文件ss.txt
    a)获取文件大小fs.sizeOf(ss.txt);
    b)计算切片大小
    Math.max(minSize, Math.min(maxSize, blockSize))=
    computeSliteSize(Math.max(minSize,Math.min(maxSize,blocksize)))=blocksize=128M

    当文件>128M的时候按上面的计算;

    当都是小文件的时候,因为一个文件分一个块,所以有几个块就有几个切片,就有几个MapTask。

    c)默认情况下,切片大小=blocksize
    d)开始切,形成第1个切片:ss.txt—0:128M 第2个切片ss.txt—128:256M 第3个切片ss.txt—256M:300M(每次切片时,都要判断切完剩下的部分是否大于块的1.1倍,不大于1.1倍就划分到上一块切片
    e)将切片信息写到一个切片规划文件中
    f)整个切片的核心过程在getSplit()方法中完成。
    g)数据切片只是在逻辑上对输入数据进行分片,并不会再磁盘上将其切分成分片进行存储。InputSplit只记录了分片的元数据信息,比如起始位置、长度以及所在的节点列表等。
    注意:block是HDFS物理上存储的数据,切片是对数据逻辑上的划分。

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

3.1.2.2 TextInputFormat

是默认的InputFormat。每条记录是一行输入。键K是LongWritable类型,存储该行在整个文件中的字节偏移量。V值是这行的内容,不包括任何行终止符(换行符和回车符)。

例如,文本中数据如下:
Rich learning form
Intelligent learning engine
Learning more convenient
From the real demand for more close to the enterprise

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

(0,Rich learning form)

(20,Intelligent learning engine)

(49,Learning more convenient)

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

很明显,键并不是行号。一般情况下,很难取得行号,因为文件按字节而不是按行切分为分片。

计算公式:字符个数+符号+换行符

3.1.2.3 CombineTextInputFormat

关于大量小文件的优化策略

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

  • 优化策略

    • 最好的办法,在数据处理系统的最前端(预处理/采集),将小文件先合并成大文件,再上传到HDFS做后续分析。
    • 补救措施:如果已经是大量小文件在HDFS中了,可以使用另一种InputFormat来做切片——CombineTextInputFormat,它的切片逻辑跟TextFileInputFormat不同:它可以将多个小文件从逻辑上规划到一个切片中,这样,多个小文件就可以交给一个maptask。
  • 具体实现步骤

    //  如果不设置InputFormat,它默认用的是TextInputFormat.class
    job.setInputFormatClass(CombineTextInputFormat.class)
    CombineTextInputFormat.setMaxInputSplitSize(job, 3145728);// 3M
    CombineTextInputFormat.setMinInputSplitSize(job, 2097152);// 2M
    

    注:在看number of splits时,和最大值(MaxSplitSize)有关、总体规律就是和低于最大值是一片、高于最大值1.5倍+,则为两片;高于最大值2倍以上则向下取整,比如文件大小65MB,切片最大值为4MB,那么切片为16个.总体来说,切片差值不超过1个,不影响整体性能。eg:文件大小是10M,10 = 3 + 3 + 3 + 1,1不够最小切片,所以分到上一个切片,变成3 + 3 + 4,3个片。再比如文件大小为11M,则能分4个片(11 = 3 + 3 + 3 + 2)。

3.1.2.4 KeyValueTextInputFormat

每一行均为一条记录,被分隔符分割为key,value。可以通过在驱动类中设置:

// 来设定分隔符。默认分隔符是tab(\t)
conf.set(KeyValueLineRecordReader.KEY_VALUE_SEPERATOR, " ");
job.setInputFormatClass(KeyValueTextInputFormat.class);

例如,文本中数据如下:
1,java hadoop hdfs
2,spark scala

逗号前面是key,后面是value

3.1.2.5 NLineInputFormat

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

Job.setInputFormatClass(NLineInputFormat.class);
// 如果不设置setNumLinesPerSplit,默认一行一个切片。
NLineInputFormat.setNumLinesPerSplit(job, 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)

(19,Intelligent learning engine)

另一个 mapper 则收到后两行:

(47,Learning more convenient)

(72,From the real demand for more close to the enterprise)

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

3.2 OutputFormat数据输出

OutputFormat是MapReduce输出的基类,所有实现MapReduce输出都实现了OutputFormat接口。

常见的OutputFormat实现类:

  • TextOutputFormat 文本输出

    默认的输出格式是TextOutputFormat,它把每条记录写为文本行。它的键和值可以是任意类型,因为TextOutputFormat调用toString()方法把它们转换为字符串。

  • SequenceFileOutputFormat

    SequenceFileOutputFormat将它的输出写为一个顺序文件。如果输出需要作为后续 MapReduce任务的输入,这便是一种好的输出格式,因为它的格式紧凑,很容易被压缩。

  • 自定义OutputFormat

    自定义一个类继承FileOutputFormat,改写recordwriter,具体改写输出数据的方法write()。

4. Join多种应用

4.1 Map join(Distributedcache分布式缓存)

  • 使用场景

    一张表十分小、一张表很大。

  • 解决方案
    在map端缓存多张表,提前处理业务逻辑,这样增加map端业务,减少reduce端数据的压力,尽可能的减少数据倾斜。

  • 具体办法:采用distributedcache

  1. 在mapper的setup阶段,将文件读取到缓存集合中。

  2. 在驱动函数中加载缓存。

    // 缓存普通文件到task运行节点
    job.addCacheFile(new URI("file:/e:/mapjoincache/pd.txt"));
    

4.2 Reduce join

  • 原理:
    Map端的主要工作:为来自不同表(文件)的key/value对打标签以区别不同来源的记录。然后用连接字段作为key,其余部分和新加的标志作为value,最后进行输出。
    Reduce端的主要工作:在reduce端以连接字段作为key的分组已经完成,我们只需要在每一个分组当中将那些来源于不同文件的记录(在map阶段已经打标志)分开,最后进行合并就ok了。
  • 该方法的缺点
    这种方式的缺点很明显就是会造成map和reduce端也就是shuffle阶段出现大量的数据传输,效率很低。

5. 数据清洗(ETL)

在运行核心业务Mapreduce程序之前,往往要先对数据进行清洗,清理掉不符合用户要求的数据。清理的过程往往只需要运行mapper程序,不需要运行reduce程序。

6. 计数器应用

Hadoop为每个作业维护若干内置计数器,以描述多项指标。例如,某些计数器记录已处理的字节数和记录数,使用户可监控已处理的输入数据量和已产生的输出数据量。
API

  • 采用枚举的方式统计计数
    enum MyCounter{MALFORORMED,NORMAL}
    //对枚举定义的自定义计数器加1
    context.getCounter(MyCounter.MALFORORMED).increment(1);
  • 采用计数器组、计数器名称的方式统计
    context.getCounter(“counterGroup”, “countera”).increment(1);
    组名和计数器名称随便起,但最好有意义。
  • 计数结果在程序运行后的控制台上查看。

7. MapReduce开发总结

在编写MapReduce程序时,需要考虑的几个方面:

  1. 输入数据接口:InputFormat
    默认使用的实现类是:TextInputFormat
    TextInputFormat的功能逻辑是:一次读一行文本,然后将该行的起始偏移量作为key,行内容作为value返回。
    KeyValueTextInputFormat:每一行均为一条记录,被分隔符分割为key,value。默认分隔符是tab(\t)。
    NlineInputFormat:按照指定的行数N来划分切片。
    CombineTextInputFormat:可以把多个小文件合并成一个切片处理,提高处理效率。
    用户还可以自定义InputFormat。
  2. 逻辑处理接口:Mapper
    用户根据业务需求实现其中三个方法:map() setup() cleanup ()
  3. Partitioner分区
    有默认实现 HashPartitioner,逻辑是根据key的哈希值和numReduces来返回一个分区号;key.hashCode()&Integer.MAXVALUE % numReduces
    如果业务上有特别的需求,可以自定义分区。
  4. Comparable排序
    当我们用自定义的对象作为key来输出时,就必须要实现WritableComparable接口,重写其中的compareTo()方法。
    部分排序:对最终输出的每一个文件进行内部排序。
    全排序:对所有数据进行排序,通常只有一个Reduce。
    二次排序:排序的条件有两个。
  5. Combiner合并
    Combiner合并可以提高程序执行效率,减少IO传输。但是使用时必须不能影响原有的业务处理结果。
  6. Reduce端分组:GroupingComparator
    ReduceTask拿到输入数据(一个partition的所有数据)后,首先需要对数据进行分组,其分组的默认原则是key相同,然后对每一组kv数据调用一次reduce()方法,并且将这一组kv中的第一个kv的key作为参数传给reduce的key,将这一组数据的value的迭代器传给reduce()的values参数。
    利用上述这个机制,我们可以实现一个高效的分组取最大值的逻辑。
    自定义一个bean对象用来封装我们的数据,然后改写其compareTo方法产生倒序排序的效果。然后自定义一个GroupingComparator,将bean对象的分组逻辑改成按照我们的业务分组id来分组(比如订单号)。这样,我们要取的最大值就是reduce()方法中传进来key。
  7. 逻辑处理接口:Reducer
    用户根据业务需求实现其中三个方法:reduce() setup() cleanup ()
  8. 输出数据接口:OutputFormat
    默认实现类是TextOutputFormat,功能逻辑是:将每一个KV对向目标文本文件中输出为一行。
    SequenceFileOutputFormat将它的输出写为一个顺序文件。如果输出需要作为后续 MapReduce任务的输入,这便是一种好的输出格式,因为它的格式紧凑,很容易被压缩。
    用户还可以自定义OutputFormat。

你可能感兴趣的:(#,Hadoop,大数据)