一. 计数器概述

执行MapReduce程序的时候,控制台输出信息中通常有下面所示片段内容:
image.png
输出信息中的核心词是counters,中文叫做计数器,Hadoop内置的计数器功能收集作业的主要统计信息,可以帮助用户理解程序的运行情况,辅助用户诊断故障。

二. MapReduce内置计数器

Hadoop为每个MapReduce作业维护一些内置的计数器,这些计数器报告各种指标,例如和MapReduce程序执行中每个阶段输入输出的数据量相关的计数器,可以帮助用户进行判断程序逻辑是否生效、正确。

Hadoop内置计数器根据功能进行分组。每个组包括若干个不同的计数器,分别是:MapReduce任务计数器(Map-Reduce Framework)、文件系统计数器(File System Counters)、作业计数器(Job Counters)、输入文件任务计数器(File Input Format Counters)、输出文件计数器(File Output Format Counters)。

需要注意的是,内置的计数器都是MapReduce程序中全局的计数器,跟MapReduce分布式运算没有关系,不是所谓的每个局部的统计信息。

1. Map-Reduce Framework Counters

计数器名称说明
MAP_INPUT_RECORDS所有mapper已处理的输入记录数
MAP_OUTPUT_RECORDS所有mapper产生的输出记录数
MAP_OUTPUT_BYTES所有mapper产生的未经压缩的输出数据的字节数
MAP_OUTPUT_MATERIALIZED_BYTESmapper输出后确实写到磁盘上字节数
COMBINE_INPUT_RECORDS所有combiner(如果有)已处理的输入记录数
COMBINE_OUTPUT_RECORDS所有combiner(如果有)已产生的输出记录数
REDUCE_INPUT_GROUPS所有reducer已处理分组的个数
REDUCE_INPUT_RECORDS所有reducer已经处理的输入记录的个数。每当某个reducer的迭代器读一个值时,该计数器的值增加
REDUCE_OUTPUT_RECORDS所有reducer输出记录数
REDUCE_SHUFFLE_BYTESShuffle时复制到reducer的字节数
SPILLED_RECORDS所有map和reduce任务溢出到磁盘的记录数
CPU_MILLISECONDS一个任务的总CPU时间,以毫秒为单位,可由/proc/cpuinfo获取
PHYSICAL_MEMORY_BYTES一个任务所用的物理内存,以字节数为单位,可由/proc/meminfo获取
VIRTUAL_MEMORY_BYTES一个任务所用虚拟内存的字节数,由/proc/meminfo获取

2. File System Counters Counters

文件系统的计数器会针对不同的文件系统使用情况进行统计,比如HDFS、本地文件系统:
image.png

计数器名称说明
BYTES_READ程序从文件系统中读取的字节数
BYTES_WRITTEN程序往文件系统中写入的字节数
READ_OPS文件系统中进行的读操作的数量(例如,open操作,filestatus操作)
LARGE_READ_OPS文件系统中进行的大规模读操作的数量
WRITE_OPS文件系统中进行的写操作的数量(例如,create操作,append操作)

3. Job Counters

image.png

计数器名称说明
Launched map tasks启动的map任务数,包括以“推测执行”方式启动的任务
Launched reduce tasks启动的reduce任务数,包括以“推测执行”方式启动的任务
Data-local map tasks与输人数据在同一节点上的map任务数
Total time spent by all maps in occupied slots (ms)所有map任务在占用的插槽中花费的总时间(毫秒)
Total time spent by all reduces in occupied slots (ms)所有reduce任务在占用的插槽中花费的总时间(毫秒)
Total time spent by all map tasks (ms)所有map task花费的时间
Total time spent by all reduce tasks (ms)所有reduce task花费的时间

4. File Input|Output Format Counters

image.png

计数器名称说明
读取的字节数(BYTES_READ)由map任务通过FilelnputFormat读取的字节数
写的字节数(BYTES_WRITTEN)由map任务(针对仅含map的作业)或者reduce任务通过FileOutputFormat写的字节数

3. MapReduce 自定义计数器

虽然Hadoop内置的计数器比较全面,给作业运行过程的监控带了方便,但是对于一些业务中的特定要求(统计过程中对某种情况发生进行计数统计)MapReduce还是提供了用户编写自定义计数器的方法。最重要的是,计数器是全局的统计,避免了用户自己维护全局变量的不利性。
自定义计数器的使用分为两步:

  1. 通过context.getCounter方法获取一个全局计数器,创建的时候需要指定计数器所属的组名和计数器的名字:
    image.png
  2. 在程序中需要使用计数器的地方,调用counter提供的方法即可,比如+1操作:
    image.png

4. 案例:MapReduce自定义计数器使用

1. 需求

针对一批文件进行词频统计,不知何种原因,在任意文件的任意地方都有可能插入单词”apple”,现要求使用计数器统计出数据中apple出现的次数,便于用户执行程序时判断。
image.png

2. 代码实现

1. Mapper类

public class WordCountMapper extends Mapper<LongWritable, Text,Text,LongWritable> {
    @Override
    protected void map(LongWritable key, Text value, Context context) throws IOException, InterruptedException {
        //从程序上下文对象获取一个全局计数器:用于统计apple出现的个数
        //需要指定计数器组 和计数器的名字
        Counter counter = context.getCounter("itcast_counters", "apple Counter");

        String[] words = value.toString().split("\\s+");
        for (String word : words) {
            //判断读取内容是否为apple  如果是 计数器加1
            if("apple".equals(word)){
                counter.increment(1);
            }
            context.write(new Text(word),new LongWritable(1));
        }
    }
}

2. Reducer类

public class WordCountReducer extends Reducer<Text, LongWritable,Text,LongWritable> {
    @Override
    protected void reduce(Text key, Iterable<LongWritable> values, Context context) throws IOException, InterruptedException {
        long count = 0;
        for (LongWritable value : values) {
            count +=value.get();
        }
        context.write(key,new LongWritable(count));
    }
}

3. 运行主类

public class WordCountDriver extends Configured implements Tool {

    @Override
    public int run(String[] args) throws Exception {
        // 创建作业实例
        Job job = Job.getInstance(getConf(), WordCountDriver.class.getSimpleName());
        // 设置作业驱动类
        job.setJarByClass(this.getClass());

        // 设置作业mapper reducer类
        job.setMapperClass(WordCountMapper.class);
        job.setReducerClass(WordCountReducer.class);

        // 设置作业mapper阶段输出key value数据类型
        job.setMapOutputKeyClass(Text.class);
        job.setMapOutputValueClass(LongWritable.class);

        //设置作业reducer阶段输出key value数据类型 也就是程序最终输出数据类型
        job.setOutputKeyClass(Text.class);
        job.setOutputValueClass(LongWritable.class);

        // 配置作业的输入数据路径
        FileInputFormat.addInputPath(job, new Path(args[0]));
        // 配置作业的输出数据路径
        FileOutputFormat.setOutputPath(job, new Path(args[1]));

        // 提交作业并等待执行完成
        return job.waitForCompletion(true) ? 0 : 1;

    }


    public static void main(String[] args) throws Exception {
        //配置文件对象
        Configuration conf = new Configuration();
        //使用工具类ToolRunner提交程序
        int status = ToolRunner.run(conf, new WordCountDriver(), args);
        //退出客户端程序 客户端退出状态码和MapReduce程序执行结果绑定
        System.exit(status);
    }
}

4. 执行结果

image.png


捕风
47 声望16 粉丝