pyspark中combineByKey的两种理解方法

Spark 1.6


以前一直模模糊糊的,现在搞一下比较清楚


combineByKey(createCombiner, mergeValue, mergeCombiners, numPartitions=None, partitionFunc=<function portable_hash at 0x7f1ac7340578>)
它是一个泛型函数,主要完成聚合操作,将输入RDD[(K,V)]转化为结果RDD[(K,C)]输出

在数据分析中,处理Key,Value的Pair数据是极为常见的场景,例如我们可以针对这样的数据进行分组、聚合或者将两个包含Pair数据的RDD根据key进行join。从函数的抽象层面看,这些操作具有共同的特征,都是将类型为RDD[(K,V)]的数据处理为RDD[(K,C)]。这里的V和C可以是相同类型,也可以是不同类型。
作者:LuciferTM
链接:http://www.jianshu.com/p/f3aea4480f2b
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

流程图

这里写图片描述

多谢@蒋潇亿--知乎的回答,我这里将图用自己的话再次整理下,方便理解

复现一下代码


def createCombiner(value):
    return (value,1)

def mergeValue(acc,value):
    return (acc[0]+value,acc[1]+1)

def mergeCombiners(acc1,acc2):
    return (acc1[0]+acc2[0],acc1[1]+acc2[1])

data = sc.parallelize([('coffee',1),('coffee',2),('panda',3),('coffee',9)],2)

# data.collect():[('coffee', 1), ('coffee', 2), ('panda', 3), ('coffee', 9)]
result = data.combineByKey(createCombiner,mergeValue,mergeCombiners)

print result.collect()

#------------------------------------------------------
#拓展,计算key所含value的均值,方法一,使用map
print result.map(lambda x:(x[0],float(x[1][0])/x[1][1])).collect()
# 方法二,s使用mapValues
print result.mapValues(lambda x:float(x[0])/x[1]).collect()


#[('coffee', (12, 3)), ('panda', (3, 1))]
#[('coffee', 4.0), ('panda', 3.0)]
#[('coffee', 4.0), ('panda', 3.0)]


来一波分析

第一种比较正统的方法,按照原理图来一步步推倒过程

  1. 如果pair RDD的key第一次出现,那么就用把该key下的value进行createCombiner操作,这里第一个pair RDD输出结果应该是这种形式('coffee',(1,1))这里需要强调的是,这是对value进行操作的,将其中的value进行转化。这里的例子是(1,1)第一个1是value,第二个1是出现了一次

  2. 对于key没重复的pair RDD才上上述同样操作,如果碰到同样key的了,那就转到第二步,key不变的情况下,将上一次的(1,1)当做参数传递进mergeValue,效果就是说,以前的acc[0]=1,即值是1,然后现在新的coffee传进来的值是2,即value=2,这样,就会对同key值进行累加acc[0]+value=同key下的累加值,而acc[1]=1,即统计了该key出现的次数,acc[1]+1=coffee这个key共出现的次数

  3. 经过这两步之后,先不考虑另一个分区的情况,如果只有一个分区,那么现在的结果应该是这样。第一个分区的结果[('coffee',(3,2)),('panda',(3,1))],然后第二个分区的结果同理('coffee',(9,1)),之后再对同key下的value传入mergeCombiner进行操作即可,方式同第二步类似,搞清楚谁是传进去的value


第二种方法,使用字典来模拟这个过程

# 相当于spark中的两个分区
part1 = [('coffee',1),('coffee',2),('panda',3)]
part2 = [('coffee',9)]

dict_res = {}
for part in [part1,part2]:
    for tup in part: 
        if tup[0] not in dict_res:
            dict_res[tup[0]]= {}  # 在该key下,将value构建dict
            dict_res[tup[0]]['sum'] = 0 
            dict_res[tup[0]]['times'] = 0

        dict_res[tup[0]]['sum'] += tup[1]  # sum叠加
        dict_res[tup[0]]['times'] +=1  # 次数累加


print dict_res

# {'coffee': {'sum': 12, 'times': 3}, 'panda': {'sum': 3, 'times': 1}}
# 其中coffee代表键,之后的value我又传了个dict,里面key=sum的value代表和,key=times的value代表前面的key如coffee出现的次数

将上面的式子再进化一次,使更像spark的写法


def createAndMergeValue(part):
    for tup in part: 
        if tup[0] not in dict_res:
            dict_res[tup[0]]= {}
            dict_res[tup[0]]['sum'] = 0
            dict_res[tup[0]]['times'] = 0

        dict_res[tup[0]]['sum'] += tup[1]
        dict_res[tup[0]]['times'] +=1

def  mergeCombiners(partitions):
    for part in partitions:
        createAndMergeValue(part)


    
dict_res = {}
partitions = [[('coffee',1),('coffee',2),('panda',3)],[('coffee',9)]]  # 为了表现其分区的特性,这里用了list区分分区部分
mergeCombiners(partitions)

print dict_res

{'coffee': {'sum': 12, 'times': 3}, 'panda': {'sum': 3, 'times': 1}}

致谢

@转--pyspark-combineByKey详解

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容