概况

Hadoop Streaming 是一个工具, 代替编写Java的实现类,而利用可执行程序来完成map-reduce过程。
一个最简单的程序

$HADOOP_HOME/bin/hadoop jar $HADOOP_HOME/hadoop-streaming.jar
-input myInputDirs
-output myOutputDir
-mapper /bin/cat
-reducer /bin/wc

各个shell变量,请自行配置

一般MAPPER_FILE和REDUCER_FILE都是shell中调用awk,所以如下的讨论基于awk。

一个完整的map-reduce程序有如下

InputFile --> mappers --> [combiner]-> [partitioner] --> reducers --> outputFiles

其中mapper是必须的,combiner和partitioner,reducer是可选的.
通过如下参数指定map和reduce的数目,reducer的数目可以为0.

-D mapred.map.tasks=64
-D mapred.reduce.tasks=0

下面详细说明

map

map的输入可以是多个文件或多个目录,一般以空格作为文件的分隔。支持通配符(其实是shell自动扩展而成)。
map主要的工作完成输入数据的规整。

一般情况下,我们可以通过目录名来区分多个输入源。在awk中,我们可以通过如下方式来区分源

if(match(ENVIRON["map_input_file"], "billserver") > 0)
{
#output
}

如上处理path路径中有billserver的日志

combiner

combiner一般可以当做apper之后的本地reducer,最主要的功能是减少网络传输。

可以认为我们在awk中end的部分就是一个本地的reducer。如下map-reduce的功能是统计第一个域出现的个数.

awk '{
dict[$1]++;
}
END{for(d in dict)
print d,dict[d];
}'

一般做法可以是 print $1,1,在reducer中再统计,但for相当于将多个$1,1,在本地合并了成了$1,cnt。

下面介绍通过combiner指定排序。

  -D map.output.key.field.separator=: \
-D mapred.text.key.comparator.options=-k2,2nr \
-D mapred.output.key.comparator.class=org.apache.hadoop.mapred.lib.KeyFieldBasedComparator \

第一个参数指定分隔符
第二个参数类似于sort的参数,指定排序的方式。

其实在awk中,除了combiner,在print的时候,通过管道调用sort命令也是可以完成类似的需求。

print d, dict[d] |"sort -t':' -k2nr"

partitioner

在一般情况下,map的输出结果需要分发到各个reducer中,partitioner就是控制分发的策略的。默认情况下,按照map结果的第一个域作为key(以\t分隔),某些情况下,我们需要将第一个域的一部分作为key分发到同一个reducer中。
Hadoop 提供了一个非常实用的partitioner类KeyFieldBasedPartitioner,通过配置相应的参数就可以使用。通过KeyFieldBasedPartitioner可以方便地实现二次排序。
使用方法:

-partitioner org.apache.hadoop.mapred.lib.KeyFieldBasedPartitioner

一般配合:

-D map.output.key.field.separator
-D num.key.fields.for.partition

map.output.key.field.separator指定key内部的分隔符
num.key.fields.for.partition 指定对key分出来的前几部分做partition而不是整个key,如果key的个数小于指定的partition数,则key的全部域作为key
举例:
需要统计十年中各个月份温度超过30°的天数,输入是每天每小时的温度数,很明显年和月需要作为key将相同年和月的记录输出到相同的reducer中,此处有两种方式,传统的方式将是单独将年和月作为主key,其它记录冗余输出。用partitioner的方式可以将map的输出为
year:month:map_day
这样map的输出只有一个key,再通过streaming方式提供的两个参数(其它参数暂时缺乏资料)

-D map.output.key.field.separator=: \
-D num.key.fields.for.partition=2 \
-partitioner org.apache.hadoop.mapred.lib.KeyFieldBasedPartitioner \

将key的分隔符指定为:,前两个域为主key,将相同年份和月份的记录分发到同一个桶(reducer)中。
这样就无须冗余输出了,对于大量的数据能够加快结果的产生效率。

reduce

reducer 可以有多路输出,但基于非常初级的封装,产生的reduce输出文件为part-xxxxx-X文件,其中X是A-Z的字母之一,使用方法如下
在命令行中启用多路输出

-outputformat org.apache.hadoop.mapred.lib.SuffixMultipleTextOutputFormat
#或
-outputformat org.apache.hadoop.mapred.lib.SuffixMultipleSequenceFileOutputFormat

前者对应于文本输入,后者于二进制输入
在reducer的代码中

printf("%s\t#A",some_str);

取值可以为A-Z,不支持自定义suffix。在output目录下,其生成 part-0000-A 文件
为了可读性,我们一般可以通过命令处理下即可

$HADOOP_PATH fs -mv $OUTPUT/*A $OUTPUT/url
$HADOOP_PATH fs -mv $OUTPUT/*B $OUTPUT/ip

参数传递

有时,我们需要通过参数来增强程序的可配置性。在启动命令中,通过cmdenv配置,如下

-cmdenv top_num=10 \
-cmdenv field_num=3 \

在程序中通过ENVIRON 数组来引用

awk 'BEGIN {
top_num=ENVIRON["top_num"]
} {}

cmdenv的变量作用域在map和reducer中均有效.
注释,在python中通过如下来引用

os.environ['name']

传递环境变量

HADOOP_HOME      计算节点上配置的Hadoop路径
LD_LIBRARY_PATH 计算节点上加载库文件的路径列表
PWD 当前工作目录
dfs_block_size 当前设置的HDFS文件块大小
map_input_file mapper正在处理的输入文件路径
mapred_job_id 作业ID
mapred_job_name 作业名
mapred_tip_id 当前任务的第几次重试
mapred_task_id 任务ID
mapred_task_is_map 当前任务是否为map
mapred_output_dir 计算输出路径
mapred_map_tasks 计算的map任务数
mapred_reduce_tasks计算的reduce任务数

在shell中可以直接引用

#mapper.sh
TASK_ID=$mapred_task_id
PDW=$PWD
BLOCK_SIZE=$dfs_block_size
#环境变量附着到输入数据并输出
while read line
do
echo "$TASK_ID $PDW $BLOCK_SIZE $line"
done #reducer.sh:
while read line
do
echo $line
done

参考
Hadoop Streaming 编程
Python写MapReduce

hadoop streaming 编程的更多相关文章

  1. hadoop streaming编程小demo(python版)

    大数据团队搞数据质量评测.自动化质检和监控平台是用django,MR也是通过python实现的.(后来发现有orc压缩问题,python不知道怎么解决,正在改成java版本) 这里展示一个python ...

  2. Hadoop-2.4.1学习之Streaming编程

    在之前的文章曾提到Hadoop不仅支持用Java编写的job,也支持其他语言编写的作业,比方Hadoop Streaming(shell.python)和Hadoop Pipes(c++),本篇文章将 ...

  3. Hadoop Streaming框架使用(一)

      Streaming简介 link:http://www.cnblogs.com/luchen927/archive/2012/01/16/2323448.html Streaming框架允许任何程 ...

  4. Streaming编程实例(c,c++,python等)

    1.概述 Hadoop Streaming是Hadoop提供的一个编程工具,它允许用户使用任何可执行文件或者脚本文件作为Mapper和Reducer,例如: 采用shell脚本语言中的一些命令作为ma ...

  5. Hadoop Streaming框架学习(一)

    Hadoop Streaming框架学习(一) Hadoop Streaming框架学习(一) 2013-08-19 12:32 by ATP_, 473 阅读, 3 评论, 收藏, 编辑 1.Had ...

  6. Apache Spark 2.2.0 中文文档 - Spark Streaming 编程指南 | ApacheCN

    Spark Streaming 编程指南 概述 一个入门示例 基础概念 依赖 初始化 StreamingContext Discretized Streams (DStreams)(离散化流) Inp ...

  7. 用python + hadoop streaming 编写分布式程序(一) -- 原理介绍,样例程序与本地调试

    相关随笔: Hadoop-1.0.4集群搭建笔记 用python + hadoop streaming 编写分布式程序(二) -- 在集群上运行与监控 用python + hadoop streami ...

  8. Spark Streaming编程示例

    近期也有开始研究使用spark streaming来实现流式处理.本文以流式计算word count为例,简单描述如何进行spark streaming编程. 1. 依赖的jar包 参考<分别用 ...

  9. hadoop streaming 文档

    Hadoop Streaming框架使用(一) Streaming简介 Streaming框架允许任何程序语言实现的程序在Hadoop MapReduce中使用,方便已有程序向Hadoop平台移植.因 ...

随机推荐

  1. 20-语言入门-20-Financial Management

    题目地址: http://acm.nyist.edu.cn/JudgeOnline/problem.php?pid=72    描述Larry graduated this year and fina ...

  2. sendmessage()模拟鼠标点击

    {鼠标软模拟:好处就是不会真的移动鼠标 开始按钮 坐标 x=386y=387 }sendmessage(hookHwnd,messages.WM_LBUTTONDOWN ,0,$0180017A); ...

  3. 注册表删除chrome插件

    注册表,对于绝大部分人来说,都是一个比较陌生的东西.然而,我们的几乎所有软件都会在这里出现. 就最近一次,公司给每个员工的chrome浏览器绑定的一堆插件,并且无法删除.手动删除插件文件后,重启机器又 ...

  4. 清幽傲竹实现kbmMWServer的方法(转)

    感谢竹子! 整体思路,是不用kbmMWUNIDACQuery,而是直接用uniQuery做数据查询,利用kbmMWUNIDACConnectioPool取得数据库联接,自己再建一个uniQuery对象 ...

  5. Android Monkey测试

    Monkey测试1——Monkey的使用 原文地址: http://www.douban.com/note/257029872/ (转自豆瓣,版权属于豆瓣及豆瓣网友,如有冒犯请见谅并联系我们) Mon ...

  6. hdu 3359 Kind of a Blur (高斯消元 浮点型)

    题目链接 题意: H * W (W,H <= 10) 的矩阵A的某个元素A[i][j],从它出发到其他点的曼哈顿距离小于等于D的所有值的和S[i][j]除上可达点的数目,构成了矩阵B.给定矩阵B ...

  7. kindeditor编辑器图片水印

    //upload_pic.ashx源码 <%@ webhandler Language="C#" class="edit_html_upload_pic" ...

  8. 转:ASP.NET MVC中Unobtrusive Ajax的妙用

    Unobtrusive Javascript有三层含义:一是在HTML代码中不会随意的插入Javsscript代码,只在标签中加一些额外的属性值,然后被引用的脚本文件识别和处理:二是通过脚本文件所增加 ...

  9. 图表框架HelloCharts(2)柱状图

    1.效果图 2.xml代码 activity_column_chart.xml <FrameLayout xmlns:android="http://schemas.android.c ...

  10. 转载RabbitMQ入门(4)--路由

    路由 (使用Java客户端) 在先前的指南中,我们建立了一个简单的日志系统.我们可以将我们的日志信息广播到多个接收者. 在这部分的指南中,我们将要往其中添加一个功能-让仅仅订阅一个消息的子集成为可能. ...