zoukankan      html  css  js  c++  java
  • Spark大数据处理 之 RDD粗粒度转换的威力

    从WordCount看Spark大数据处理的核心机制(2)中我们看到Spark为了支持迭代和交互式数据挖掘,而明确提出了内存中可重用的数据集RDD。RDD的只读特性,再加上粗粒度转换操作形成的Lineage,形成了它独立的高效容错机制。

    RDD的粗粒度的转换是否有足够的表达能力,来支持多种多样的应用需求呢?先看看RDD究竟有哪些API,然后看它们如何模拟Google经典的MapReduce和图数据处理框架Pregel。

    RDD的API

    转换

    def map[U](f: T => U): RDD[U]
    

    将RDD[T]经过f转换成RDD[U],T和U一一映射,两个RDD元素个数相等

    def flatMap[U](f: T => TraversableOnce[U]): RDD[U]
    

    将RDD[T]经过f闭包转换成RDD[U],一个T可以映射成0到多个U,两个RDD元素通常不等

    def mapPartitions[U: ClassTag](
          f: Iterator[T] => Iterator[U]): RDD[U]
    

    mapPartitions是partition级的转换,多元素到多元素或单元素的转换。

    还记得从WordCount看Spark大数据处理的核心机制(1)中我们扒开的countByValue函数吗?它就是通过mapPartitions来统计每个partition上所有单词的计数。

    def union(other: RDD[T]): RDD[T]
    

    将两个RDD[T]合并成一个RDD[T]。

    可能一开始会觉得union操作会耗时较大,实际上这个操作非常廉价。RDD的元信息中包含了Partition/Lineage等信息,union只是合并元信息,而并不涉及具体的数据,so easy。

    def distinct(): RDD[T]
    

    将原RDD[T]转换成新的RDD[T],但每个元素只出现一次。

    distinct从业务意义上很容易理解,但消耗却不少,需要通过网络交换各个Partition的数据,小伙伴们要注意了。

    以下所有的转换都仅针对RDD[(K, V)]有效,是通过把RDD[(K, V)]隐式转换成PairRDDFunctions[K, V]获得的。使用前一定要导入SparkContext内的隐式转化函数,如下:

    import org.apache.spark.SparkContext._
    

    不然找不到下面的函数,不要说一码不负责乱说哈。

    隐式转换是Scala带来的好东西,类似于C#或Ruby中可以把类打开的功能,实在是写出优雅代码不可多得的工具。不清楚的小伙伴记得一定要GFSOSO哈。

    def groupByKey(): RDD[(K, Iterable[V])]
    

    将RDD[(K, V)]中所有(K, V)键值对按K进行分组,每组一个元素,形成新的RDD[(K, Iterable[V])]。

    def reduceByKey(func: (V, V) => V): RDD[(K, V)]
    

    将RDD[(K, V)]中所有(K, V)键值对按K进行分组归并,最终每个K只有一个(K, V)与之对应。

    def join[W](other: RDD[(K, W)]): RDD[(K, (V, W))]
    

    将RDD[(K, V)]与RDD[(K, W)做join操作,形成新的RDD[(K, (V, W)),最终形成的RDD中只有两个RDD中共有的K。

    def cogroup[W](other: RDD[(K, W)]): RDD[(K, (Iterable[V], Iterable[W]))]
    

    将RDD[(K, V)]与RDD[(K, W)一起分组。

    值得注意的是:结果里面包含两个RDD中所有的K,也就是说Iterable[V]和Iterable[W]中可能某个为空。

    def mapValues[U](f: V => U): RDD[(K, U)]
    

    仅对(K, V)中的V进行转换,转换后和原先的K一起形成新的(K, U)键值对,通常和groupByKey一起使用。

    def partitionBy(partitioner: Partitioner): RDD[(K, V)]
    

    将RDD[(K, V)]按照K进行重新分区。

    重新分区对每个分区数据非常少的情况很有帮助。减少分区,任务数量也会随着分区的减少而减少,降低大量任务调度的开销。

    所有转换返回的一定是RDD。

    动作

    def count(): Long
    

    统计RDD[T]中元素T的个数。

    def reduce(f: (T, T) => T): T
    

    对RDD[T]中的元素进行两两合并,最终合并成一个值。

    比如对RDD[Int]中的所有元素求和:ints.reduce((i1, i2) => i1 + i2)

    def collect(): Array[T]
    

    将RDD[T]中所有元素从Slave上收集到Driver上。

    该方法应该只应用在元素较少的RDD上,否则Driver一定OutOfMemory。

    def saveAsTextFile(path: String): Unit
    

    把RDD[T]中的所有元素保存到磁盘,通常是保存到分布式文件系统。

    注意path是个目录,RDD中的每个元素对应了目录下的一个文件。保存后形成:/path/part-00000,
    /path/part-00001等文件。最大的好处就是解决了通过collect到Driver再保存到磁盘的问题。

    常用的RDD API就上面这些,更多请参考Spark官方文档。

    所有动作返回的一定不是RDD。

    转换不会加载数据,仅记录Lineage而已,而动作会触发数据的加载,并根据Lineage完成所有的转换,是延迟计算,极大地提升效率。

    RDD对MapReduce的模拟

    来看这篇文章的小伙伴们应该都清楚MapReduce模型了,它很容易使用RDD进行描述。假设有一个输入数据集(其元素类型为T),和两个函数myMap: T => List[(Ki, Vi)] 和 myReduce: (Ki; List[Vi]) ) List[R],RDD API模拟代码如下:

    data.flatMap(myMap)
    	.groupByKey()
    	.map((k, vs) => myReduce(k, vs))
    

    如果任务包含combiner,则相应的代码为:

    data.flatMap(myMap)
    	.reduceByKey(myCombiner)
    	.map((k, v) => myReduce(k, v))
    

    RDD对Pregel图计算的模拟

    Pregel是面向图算法的基于BSP范式的编程模型。程序由一系列超步(Superstep)协调迭代运行。在每个超步中,各个顶点执行用户函数,并更新相应的顶点状态,变异图拓扑,然后向下一个超步的顶点集发送消息。这种模型能够描述很多图算法,包括最短路径,双边匹配和PageRank等,我们以PageRank为例来说明。

    PageRank可是搜索引擎的基础,经典的大数据算法,还不知道的小伙伴请自行GFSOSO哈。

    当前PageRank记为r,顶点表示状态。在每个超步中,各个顶点向其所有邻居发送贡献值r/n,这里n是邻居的数目。下一个超步开始时,每个顶点将其分值(rank)更新为 α/N + (1 - α) * Σci,这里的求和是各个顶点收到的所有贡献值的和,N是顶点的总数。

    Pregel的通信模式可以用RDD来描述,主要思想是:将每个超步中的顶点状态和要发送的消息存储为RDD,然后根据顶点ID分组,进行Shuffle通信(即cogroup操作)。然后对每个顶点ID上的状态和消息应用用户函数(即mapValues操作),产生一个新的RDD,即(VertexID, (NewState, OutgoingMessages))。然后执行map操作分离出下一次迭代的顶点状态和消息(即mapValues和flatMap操作)。代码如下:

    val vertices = // RDD of (ID, Vertice) pairs
    val incomingMessages = // RDD of (ID, Message) pairs
    val grouped = vertices.cogroup(incomingMessages)
    val newData = grouped.mapValues {
        (vert, incomingMsgs) => spreadRank(vert, incomingMsgs)
        // returns (newState, outgoingMsgs)
    }.cache()
    val newVerts = newData.mapValues((v,ms) => v)
    val newMsgs = newData.flatMap((id,(v,ms)) => ms)
    
    def spreadRank(...): ... = {
    	// spread the incoming rank to outgoing rank
    }
    

    需要注意的是,这种实现方法中,RDD grouped,newData和newVerts的分区方法与输入RDD vertices一样。所以,顶点状态一直存在于它们开始执行的机器上,这跟原Pregel一样,这样就减少了通信成本。因为cogroup和mapValues保持了与输入RDD相同的分区方法,所以分区是自动进行的。

    如果觉得上面这一段有难度,请在微信公众号上联系一码。

    经过四篇文章,Spark基础知识还剩下共享变量,下一篇文章讲过共享变量后,开始讲开发Spark应用经常遇到的问题,以及如何优化性能。

    推荐

    动手写Count

    从WordCount看Spark大数据处理的核心机制(1)

    从WordCount看Spark大数据处理的核心机制(2)

    RDD粗粒度转换的威力

    查看《Spark大数据处理》系列文章,请进入YoyaProgrammer公众号,点击 核心技术,点击 Spark大数据处理。

    分类 Spark大数据处理

    优雅程序员 原创 转载请注明出处

    图片二维码

  • 相关阅读:
    C#中利用iTextSharp开发二维码防伪标签(1)
    delphi 数据库中Connection与Query连接数量问题思考
    cPanel 安装方法
    招商行用卡人工服务方式
    软链接的创建和查看
    zencart低版本由php5.2.17升级PHP5.3环境下错误及解决方案
    EXCEL应用:高级筛选里的条件或和与的条件怎么写 例:不包含,包含等
    array_walk与array_map 的不同 array_filter
    zen cart global $db 这噶哒
    hdu 5655 CA Loves Stick
  • 原文地址:https://www.cnblogs.com/yoyaprogrammer/p/rdd_api.html
Copyright © 2011-2022 走看看