RDD是什么?
RDD是Spark中的抽象数据结构类型,任何数据在Spark中都被表示为RDD。从编程的角度来看,RDD可以简单看成是一个数组。和普通数组的区别是,RDD中的数据是分区存储的,这样不同分区的数据就可以分布在不同的机器上,同时可以被并行处理。因此,Spark应用程序所做的无非是把需要处理的数据转换为RDD,然后对RDD进行一系列的变换和操作从而得到结果。本文为第一部分,将介绍Spark RDD中与Map和Reduce相关的API中。
如何创建RDD?
RDD可以从普通数组创建出来,也可以从文件系统或者HDFS中的文件创建出来。
举例:从普通数组创建RDD,里面包含了1到9这9个数字,它们分别在3个分区中。
scala> val a = sc.parallelize(1 to 9, 3)
a: org.apache.spark.rdd.RDD[Int] = ParallelCollectionRDD[1] at parallelize at <console>:12
举例:读取文件README.md来创建RDD,文件中的每一行就是RDD中的一个元素
scala> val b = sc.textFile("README.md")
b: org.apache.spark.rdd.RDD[String] = MappedRDD[3] at textFile at <console>:12
虽然还有别的方式可以创建RDD,但在本文中我们主要使用上述两种方式来创建RDD以说明RDD的API。
map
map是对RDD中的每个元素都执行一个指定的函数来产生一个新的RDD。任何原RDD中的元素在新RDD中都有且只有一个元素与之对应。
举例:
scala> val a = sc.parallelize(1 to 9, 3)
scala> val b = a.map(x => x*2)
scala> a.collect
res10: Array[Int] = Array(1, 2, 3, 4, 5, 6, 7, 8, 9)
scala> b.collect
res11: Array[Int] = Array(2, 4, 6, 8, 10, 12, 14, 16, 18)
上述例子中把原RDD中每个元素都乘以2来产生一个新的RDD。
mapPartitions
mapPartitions是map的一个变种。map的输入函数是应用于RDD中每个元素,而mapPartitions的输入函数是应用于每个分区,也就是把每个分区中的内容作为整体来处理的。
它的函数定义为:
def mapPartitions[U: ClassTag](f: Iterator[T] => Iterator[U], preservesPartitioning: Boolean = false): RDD[U]
f即为输入函数,它处理每个分区里面的内容。每个分区中的内容将以Iterator[T]传递给输入函数f,f的输出结果是Iterator[U]。最终的RDD由所有分区经过输入函数处理后的结果合并起来的。
举例:
scala> val a = sc.parallelize(1 to 9, 3)
scala> def myfunc[T](iter: Iterator[T]) : Iterator[(T, T)] = {
var res = List[(T, T)]()
var pre = iter.next while (iter.hasNext) {
val cur = iter.next;
res .::= (pre, cur) pre = cur;
}
res.iterator
}
scala> a.mapPartitions(myfunc).collect
res0: Array[(Int, Int)] = Array((2,3), (1,2), (5,6), (4,5), (8,9), (7,8))
上述例子中的函数myfunc是把分区中一个元素和它的下一个元素组成一个Tuple。因为分区中最后一个元素没有下一个元素了,所以(3,4)和(6,7)不在结果中。
mapPartitions还有些变种,比如mapPartitionsWithContext,它能把处理过程中的一些状态信息传递给用户指定的输入函数。还有mapPartitionsWithIndex,它能把分区的index传递给用户指定的输入函数。
mapValues
mapValues顾名思义就是输入函数应用于RDD中Kev-Value的Value,原RDD中的Key保持不变,与新的Value一起组成新的RDD中的元素。因此,该函数只适用于元素为KV对的RDD。
举例:
scala> val a = sc.parallelize(List("dog", "tiger", "lion", "cat", "panther", " eagle"), 2)
scala> val b = a.map(x => (x.length, x))
scala> b.mapValues("x" + _ + "x").collect
res5: Array[(Int, String)] = Array((3,xdogx), (5,xtigerx), (4,xlionx),(3,xcatx), (7,xpantherx), (5,xeaglex))
mapWith
mapWith是map的另外一个变种,map只需要一个输入函数,而mapWith有两个输入函数。它的定义如下:
def mapWith[A: ClassTag, U: ](constructA: Int => A, preservesPartitioning: Boolean = false)(f: (T, A) => U): RDD[U]
第一个函数constructA是把RDD的partition index(index从0开始)作为输入,输出为新类型A;
第二个函数f是把二元组(T, A)作为输入(其中T为原RDD中的元素,A为第一个函数的输出),输出类型为U。
举例:把partition index 乘以10,然后加上2作为新的RDD的元素。
val x = sc.parallelize(List(1,2,3,4,5,6,7,8,9,10), 3)
x.mapWith(a => a * 10)((a, b) => (b + 2)).collect
res4: Array[Int] = Array(2, 2, 2, 12, 12, 12, 22, 22, 22, 22)
flatMap
与map类似,区别是原RDD中的元素经map处理后只能生成一个元素,而原RDD中的元素经flatmap处理后可生成多个元素来构建新RDD。
举例:对原RDD中的每个元素x产生y个元素(从1到y,y为元素x的值)
scala> val a = sc.parallelize(1 to 4, 2)
scala> val b = a.flatMap(x => 1 to x)
scala> b.collect
res12: Array[Int] = Array(1, 1, 2, 1, 2, 3, 1, 2, 3, 4)
flatMapWith
flatMapWith与mapWith很类似,都是接收两个函数,一个函数把partitionIndex作为输入,输出是一个新类型A;另外一个函数是以二元组(T,A)作为输入,输出为一个序列,这些序列里面的元素组成了新的RDD。它的定义如下:
def flatMapWith[A: ClassTag, U: ClassTag](constructA: Int => A, preservesPartitioning: Boolean = false)(f: (T, A) => Seq[U]): RDD[U]
举例:
scala> val a = sc.parallelize(List(1,2,3,4,5,6,7,8,9), 3)
scala> a.flatMapWith(x => x, true)((x, y) => List(y, x)).collect
res58: Array[Int] = Array(0, 1, 0, 2, 0, 3, 1, 4, 1, 5, 1, 6, 2, 7, 2,
8, 2, 9)
flatMapValues
flatMapValues类似于mapValues,不同的在于flatMapValues应用于元素为KV对的RDD中Value。每个一元素的Value被输入函数映射为一系列的值,然后这些值再与原RDD中的Key组成一系列新的KV对。
举例
scala> val a = sc.parallelize(List((1,2),(3,4),(3,6)))
scala> val b = a.flatMapValues(x=>x.to(5))
scala> b.collect
res3: Array[(Int, Int)] = Array((1,2), (1,3), (1,4), (1,5), (3,4), (3,5))
上述例子中原RDD中每个元素的值被转换为一个序列(从其当前值到5),比如第一个KV对(1,2), 其值2被转换为2,3,4,5。然后其再与原KV对中Key组成一系列新的KV对(1,2),(1,3),(1,4),(1,5)。
reduce
reduce将RDD中元素两两传递给输入函数,同时产生一个新的值,新产生的值与RDD中下一个元素再被传递给输入函数直到最后只有一个值为止。
举例
scala> val c = sc.parallelize(1 to 10)
scala> c.reduce((x, y) => x + y)
res4: Int = 55
上述例子对RDD中的元素求和。
reduceByKey
顾名思义,reduceByKey就是对元素为KV对的RDD中Key相同的元素的Value进行reduce,因此,Key相同的多个元素的值被reduce为一个值,然后与原RDD中的Key组成一个新的KV对。
举例:
scala> val a = sc.parallelize(List((1,2),(3,4),(3,6)))
scala> a.reduceByKey((x,y) => x + y).collect
res7: Array[(Int, Int)] = Array((1,2), (3,10))
上述例子中,对Key相同的元素的值求和,因此Key为3的两个元素被转为了(3,10)。
分享到:
相关推荐
### Spark RDD详解 #### Spark计算模型与RDD概念 在探讨Spark的弹性分布式数据集(RDD)之前,我们首先需要理解Spark的基本计算模型。Spark是一种基于内存的分布式计算框架,其核心设计思想在于通过缓存中间结果来...
通过本次实验,不仅熟悉了Spark中RDD的基本操作,包括`map`、`filter`、`reduceByKey`等,还掌握了如何利用这些操作来解决实际问题。此外,还学习了如何使用累加器来统计特定信息。整个实验过程加深了对Spark编程的...
Apache Spark不仅提供了一个高效的并行计算框架,还通过其丰富的API和灵活的编程接口使得开发者能够轻松应对大规模数据处理的需求。随着大数据技术的不断发展,Spark将在更多领域发挥重要作用。
**Spark 1.0.0 API(Scala)详解** Spark是一个快速、通用且可扩展的大数据处理框架,它提供了一套强大的API,使得开发者能够高效地处理大规模数据。在Spark 1.0.0版本中,它引入了对Scala语言的深度支持,使得开发...
### RDD编程API详解 #### 一、概述 在Apache Spark框架中,弹性分布式数据集(Resilient Distributed Dataset,简称RDD)是基本的数据抽象。它是一个不可变的、分布式的对象集合,可以并行地作用于集群上的节点。...
《Spark SQL与RDD API详解》 Spark作为大数据处理领域的重要工具,其强大的处理能力和灵活性深受开发者喜爱。在Spark中,SQL和RDD(弹性分布式数据集)API是两个核心的组件,它们提供了不同的数据处理方式,使得...
MapReduce则是并行处理数据的计算框架,它将大型任务拆分成小的Map和Reduce阶段,以便在集群中并行执行。 Zookeeper是另一个Hadoop生态系统中的关键组件,它是一个分布式协调服务,用于管理配置信息、命名服务、...
4. **数据处理**:应用RDD转换和行动,如`map()`、`filter()`、`reduce()`等,构建数据处理逻辑。 5. **运行和监控**:提交你的Spark程序到集群,使用Spark UI或YARN Web UI监控任务执行情况。 五、SparkCore与Java...
《Spark详解:RDD的核心与应用》 Spark,作为大数据处理领域的明星框架,以其高效、灵活和易用的特点,迅速在大数据处理领域占据了重要的位置。本章将深入探讨Spark的核心组件——弹性分布式数据集(Resilient ...
### RDD使用基础详解 #### 一、RDD简介与特点 **RDD**(Resilient Distributed Dataset)是Apache Spark的核心概念之一,它代表一种只读的、可分区的分布式数据集。RDD具有高度的容错性,并能够透明地利用内存来...
#### 六、RDD详解 - **概念**:RDD是Spark中的核心数据结构,它是一种不可变的、分布式的、容错的数据集合。 - **特性**: - **分布性**:RDD可以在多台机器上进行并行计算。 - **弹性**:当内存不足时,RDD会...
**Spark上的WordCount程序详解** Spark作为一个分布式计算框架,提供了高效、灵活的数据处理能力。在Spark上实现WordCount是入门的经典示例,它演示了如何处理大规模文本数据并进行简单的统计分析。在这个程序中,...
### Spark的常用操作详解 #### 一、概述 Apache Spark 是一种快速且通用的大规模数据处理引擎,它支持多种计算模式,包括批处理、实时数据流处理、机器学习和图形处理等。Spark 提供了一个高度统一的编程模型,...
- **常见RDD算子**:包括`map`、`filter`、`reduceByKey`等,这些算子可以帮助用户进行数据的转换和聚合操作。 - `map`:对RDD中的每个元素应用一个函数,并返回一个新的RDD。 - `filter`:选择满足特定条件的元素...
RDD通过一系列转换操作(如map、filter等)和动作操作(如count、collect等)来处理数据。 - **转换操作**:不会立即执行,而是将操作记录下来,形成计算依赖图。 - **动作操作**:触发实际计算过程,执行前面记录...
MapReduce是Google提出的一种分布式计算模型,它将复杂的并行计算任务分解为两个主要阶段:Map(映射)和Reduce(规约)。Map阶段负责将原始输入数据分割成多个键值对,并分别处理;Reduce阶段则负责对Map阶段产生的...
DataFrame是基于Spark SQL的抽象,提供了DataFrame API,而Dataset是DataFrame的类型安全版本,结合了RDD的灵活性和DataFrame的优化。 **五、Spark Streaming** Spark Streaming处理实时数据流,通过微批处理实现...