本文参考

参考《Spark快速大数据分析》动物书中的第三章"RDD编程",前一篇文章已经概述了转化操作相关的API,本文再介绍行动操作API

和转化操作API不同的是,行动操作API只能作用于一个RDD

RDD转化操作API归纳:https://www.cnblogs.com/kuluo/p/12545374.html

Pair RDD转化操作API归纳:https://www.cnblogs.com/kuluo/p/12558563.html

Pair RDD行动操作API归纳:https://www.cnblogs.com/kuluo/p/12567221.html

环境

idea + spark 2.4.5 + scala 2.11.12

RDD均通过SparkContext的parallelize()函数创建

collect()函数

目的:

返回RDD中的所有元素

代码:

val testList = List(1, 2, 3, 3)
val testRdd = sc.parallelize(testList)
testRdd.collect().foreach(ele => print(s"$ele "))

输出:

1 2 3 3

注意:

该函数会将所有元素放入驱动器进程中,只有当整个数据集能在单台机器的内存中放得下时,才能使用,不适宜用在大规模数据集,因此大多数情况下用于本地开发测试,以下还会碰到很多这样不适合大数据量访问的函数

count()函数

目的:

返回RDD中的元素个数

代码:

val testList = List(1, 2, 3, 3)
val testRdd = sc.parallelize(testList)
println(testRdd.count())

输出:

4

countByValue()函数

目的:

各元素在RDD中出现的次数

在经典的WordCount示例程序中我们一般先通过flatMap()函数进行压扁,再用map()函数转为pair RDD,最后用reduceByKey()函数进行计数,实际上我们可以通过countByValue()函数,将WordCount示例程序中的map()函数和reduceByKey()函数两步操作,合并为一步

代码:

val testList = List(1, 2, 3, 3)
val testRdd = sc.parallelize(testList)
testRdd.countByValue().foreach(ele => print(s"$ele "))

输出:

(1,1) (3,2) (2,1)

和WordCount程序的对比:

val testList = List("a a a a b b c c", "a a c c c d d", "b b c c d", "a b c d")
val testRdd = sc.parallelize(testList)
testRdd.flatMap(line =>

line.split(" ")
).map(ele =>

  (ele, 1)
).reduceByKey{case (x, y) =>

  x + y
}.foreach(pair => print(s"$pair "))

使用countByValue后

val testList = List("a a a a b b c c", "a a c c c d d", "b b c c d", "a b c d")
val testRdd = sc.parallelize(testList)
testRdd.flatMap(line =>

  line.split(" ")
).countByValue().foreach(pair => print(s"$pair "))

两个示例程序输出结果相同:(d,4) (a,7) (b,5) (c,8)

注意:

尽管使用该函数能够简化我们的编程,但是在源码中有如下注释:

This method should only be used if the resulting map is expected to be small, as the whole thing is loaded into the driver's memory.

To handle very large results, consider using rdd.map(x => (x, 1L)).reduceByKey(_ + _) , which returns an RDD[T, Long] instead of a map.

由此可见countByValue()函数也会将所有元素加载到驱动器进程,建议在数据集不大的情况下使用

另外一点要注意的是,countByValue()函数返回的RDD中的每个元素的类型是Map,但并不是我们常说的pair RDD,pair RDD相关的转化和行动操作不能应用在元素类型是Map的RDD上

take()函数

目的:

从RDD中返回num个元素,将尽可能访问少的分区

代码:

val testList = List(3, 4, 2, 1, 9, 6, 3, 6, 8, 9)
val testRdd = sc.parallelize(testList)
testRdd.take(5).foreach(pair => print(s"$pair "))

输出:

3 4 2 1 9(不一定)

注意:

This method should only be used if the resulting array is expected to be small, as all the data is loaded into the driver's memory.

take ()函数也会将所有元素加载到驱动器进程,建议在返回元素不多的情况下使用

takeSample()函数

目的:

从RDD中返回任意一些元素

代码:

val testList = List(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
val testRdd = sc.parallelize(testList)
testRdd.takeSample(true, 5).foreach(pair => print(s"$pair "))

输出:

0 4 0 3 2(会出现重复元素)

代码:

val testList = List(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
val testRdd = sc.parallelize(testList)
testRdd. takeSample(false, 5).foreach(pair => print(s"$pair "))

输出:

0 3 8 2 7(不会出现重复元素)

sample()函数、take()函数和takeSample()函数的区别:

sample()函数是转化操作API,withReplacement参数取false时,fraction参数取[0, 1]并表示RDD中每个元素被选中的可能性

take()函数是行动操作API,从RDD中抽取由num参数指定的数量的元素,首先访问RDD中的第一个分区,若第一个分区中的元素个数已经能够满足我们num参数指定的数量,将不再访问其它分区,即尽可能少地访问RDD分区,以下为take()函数源码注释

Take the first num elements of the RDD. It works by first scanning one partition, and use the results from that partition to estimate the number of additional partitions needed to satisfy the limit.

takeSample()函数是行动操作API,从withReplacement参数取false时,结果RDD中不会出现重复元素,num参数表示要取得的元素的个数

sample()函数takeSample()函数都有第三个参数seed用来设置随机数种子,保证每次抽取RDD元素的结果集不同

take()函数和takeSample()函数被明确指明各元素会被加载到驱动器进程,不适合大数据量的情况下使用

top()函数

目的:

从RDD中返回最前面的num个元素,降序排序

内部实际上调用takeOrdered()函数

代码:

val testList = List(3, 4, 2, 1, 9, 6, 3, 6, 8, 9)
val testRdd = sc.parallelize(testList)
testRdd.top(5).foreach(pair => print(s"$pair "))

等价写法

val testList = List(3, 4, 2, 1, 9, 6, 3, 6, 8, 9)
val testRdd = sc.parallelize(testList)
testRdd.takeOrdered(5)(Ordering[Int].reverse).foreach(pair => print(s"$pair "))

输出:

9 9 8 6 6

takeOrdered()函数

目的:

从RDD中按照提供的顺序返回最前面的num个元素,默认升序排序

代码:

val testList = List(3, 4, 2, 1, 9, 6, 3, 6, 8, 9)
val testRdd = sc.parallelize(testList)
testRdd.takeOrdered(5).foreach(pair => print(s"$pair "))

输出:

1 2 3 3 4

reduce()函数

目的:

并行整合RDD中所有数据,操作两个相同元素类型的RDD数据返回一个同样类型的新元素

代码:

val testList = List(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
val testRdd = sc.parallelize(testList)
println(testRdd.reduce(_ + _))

输出:

45

fold()函数

目的:

和reduce()函数一样,但需要提供初始值,提供的初始值应当是你提供的操作的单位元素,如加法运算的单位元素是0,乘法运算的单位元素是1

代码:

val testList = List(0, 1, 2, 3, 4, 5, 6, 7, 8, 9)
val testRdd = sc.parallelize(testList)
println(testRdd.fold(0)(_ + _))

输出:

45

aggregate()函数

目的:

和reduce()函数相似,但返回值类型可以和操作的RDD类型不同,需要提供初始值,合并RDD内元素的函数以及将不同节点上的元素进行合并

代码:

val testList = List(3, 4, 5, 3, 8, 6, 6, 7, 9, 1)
val testRdd = sc.parallelize(testList)
val result:(Int, Int)
= testRdd.aggregate((0, 0))((x, y)
=> {

  (x._1 + y, x._2 + 1)
}, (x, y) => {
  (x._1 + y._1, x._2 + y._2)
})
print(1.0 * result._1 / result._2)

计算平均值,在本例中aggregate()函数返回的是一个Tuple元组

输出:

5.2

foreach()函数

目的:

对RDD中的每个元素使用给定的函数,在前面的示例中均以出现,此处不再做演示

更高效的操作:

val testList = List(3, 4, 5, 3, 8, 6, 6, 7, 9, 1)
val testRdd = sc.parallelize(testList)
testRdd.foreachPartition(partition => {

  partition.foreach(ele => {

    print(s"$ele ")
  })
})

输出:3 4 5 3 8 6 6 7 9 1

Spark学习摘记 —— RDD行动操作API归纳的更多相关文章

  1. Spark学习摘记 —— RDD转化操作API归纳

    本文参考 在阅读了<Spark快速大数据分析>动物书后,大概了解到了spark常用的api,不过书中并没有给予所有api具体的示例,而且现在spark的最新版本已经上升到了2.4.5,动物 ...

  2. Spark学习摘记 —— Pair RDD行动操作API归纳

    本文参考 参考<Spark快速大数据分析>动物书中的第四章"键值对操作",本篇是对RDD转化操作和行动操作API归纳的最后一篇 RDD转化操作API归纳:https:/ ...

  3. Spark学习摘记 —— Pair RDD转化操作API归纳

    本文参考 参考<Spark快速大数据分析>动物书中的第四章"键值对操作",由于pair RDD的一些特殊操作,没有和前面两篇的API归纳放在一起做示例 前面的几个api ...

  4. Spark学习之RDD编程(2)

    Spark学习之RDD编程(2) 1. Spark中的RDD是一个不可变的分布式对象集合. 2. 在Spark中数据的操作不外乎创建RDD.转化已有的RDD以及调用RDD操作进行求值. 3. 创建RD ...

  5. spark学习(10)-RDD的介绍和常用算子

    RDD(弹性分布式数据集,里面并不存储真正要计算的数据,你对RDD的操作,他会在Driver端转换成Task,下发到Executor计算分散在多台集群上的数据) RDD是一个代理,你对代理进行操作,他 ...

  6. Spark学习之RDD

    RDD概述 什么是RDD RDD(Resilient Distributed Dataset)叫做弹性分布式数据集,是Spark中最基本的数据抽象,它代表一个不可变.可分区.里面的元素可并行计算的集合 ...

  7. Spark学习之RDD的理解

    转自:http://www.infoq.com/cn/articles/spark-core-rdd/ 感谢张逸老师的无私分享 RDD,全称为Resilient Distributed Dataset ...

  8. Spark学习之RDD编程总结

    Spark 对数据的核心抽象——弹性分布式数据集(Resilient Distributed Dataset,简称 RDD).RDD 其实就是分布式的元素集合.在 Spark 中,对数据的所有操作不外 ...

  9. Spark学习笔记——RDD编程

    1.RDD——弹性分布式数据集(Resilient Distributed Dataset) RDD是一个分布式的元素集合,在Spark中,对数据的操作就是创建RDD.转换已有的RDD和调用RDD操作 ...

随机推荐

  1. Vue 源码解读(9)—— 编译器 之 优化

    前言 上一篇文章 Vue 源码解读(8)-- 编译器 之 解析 详细详解了编译器的第一部分,如何将 html 模版字符串编译成 AST.今天带来编译器的第二部分,优化 AST,也是大家常说的静态标记. ...

  2. 【C# 线程】C++与C#数据类型对照表

    详细:http://lzltool.com/doc/csharptocaddadd

  3. 【C#基础概念】虚方法virtual

    目录: 一.虚方法的概念 二.虚方法的特点 三.虚方法的作用 四.虚方法的声明 五.虚方法的执行 六.虚拟类的规则 一.虚方法的概念 在C#中,虚方法就是可以被子类重写的方法,如果子类重写了虚方法,则 ...

  4. java ssh远程服务器并执行多条shell命令

    java ssh远程服务器并执行多条命令 import java.io.BufferedReader; import java.io.IOException; import java.io.Input ...

  5. SQL从零到迅速精通【查询利器】

    1.[列选取]从fruits表中获取f_name和f_price两列,T-SQL语句如下. SELECT f_name,f_price FROM fruits; 2.[去重]查询fruits表中s_i ...

  6. .Net Core(.NET6)中接入Log4net和NLog进行日志记录

    一.接入Log4net 1.按日期和大小混合分割日志 nuget包安装 log4net Microsoft.Extensions.Logging.Log4Net.AspNetCore 配置文件 配置文 ...

  7. netty通信

    学习netty之前,要先了解操作系统中的IO.零拷贝(已经附上链接了) 一.netty的简单介绍 Netty 是由 JBOSS 提供的一个 Java 开源框架,现为 Github 上的独立项目. Ne ...

  8. Windows原理深入学习系列-信任等级检查

    这是[信安成长计划]的第 23 篇文章 0x00 目录 0x01 介绍 0x02 逆向分析 Win10_x64_20H2 0x03 WinDBG 0x04 参考文章 在之前的时候,一直以为 SACL ...

  9. Python:用pyinstrument做性能分析

    导引 在计算密集型计算或一些Web应用中,我们常常需要对代码做性能分析.在Python中,最原始的方法即是使用time包中的time函数(该函数以秒为计时单位): from time import s ...

  10. 详解 Java 内部类

    内部类在 Java 里面算是非常常见的一个功能了,在日常开发中我们肯定多多少少都用过,这里总结一下关于 Java 中内部类的相关知识点和一些使用内部类时需要注意的点. 从种类上说,内部类可以分为四类: ...