You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

9.0 KiB

16 | Spark StreamingSpark的实时流计算API

你好,我是蔡元楠。

今天我要与你分享的内容是“Spark Streaming”。

通过上一讲的内容我们深入了解了Spark SQL API。通过它我们可以像查询关系型数据库一样查询Spark的数据并且对原生数据做相应的转换和动作。

但是无论是DataFrame API还是DataSet API都是基于批处理模式对静态数据进行处理的。比如在每天某个特定的时间对一天的日志进行处理分析。

在第二章中你已经知道了批处理和流处理是大数据处理最常见的两个场景。那么作为当下最流行的大数据处理平台之一Spark是否支持流处理呢

答案是肯定的。

早在2013年Spark的流处理组件Spark Streaming就发布了。之后经过好几年的迭代与改进现在的Spark Streaming已经非常成熟在业界应用十分广泛。

今天就让我们一起揭开Spark Streaming的神秘面纱让它成为我们手中的利器。

Spark Streaming的原理

Spark Streaming的原理与微积分的思想很类似。

在大学的微积分课上,你的老师一定说过,微分就是无限细分,积分就是对无限细分的每一段进行求和。它本质上把一个连续的问题转换成了无限个离散的问题。

比如用微积分思想求下图中阴影部分S的面积。

我们可以把S无限细分成无数个小矩形因为矩形的宽足够短所以它顶端的边近似是一个直线。这样把容易计算的矩形面积相加就得到不容易直接计算的不规则图形面积。

你知道流处理的数据是一系列连续不断变化且无边界的。我们永远无法预测下一秒的数据是什么样。Spark Streaming用时间片拆分了无限的数据流然后对每一个数据片用类似于批处理的方法进行处理输出的数据也是一块一块的。如下图所示。

Spark Streaming提供一个对于流数据的抽象DStream。DStream可以由来自Apache Kafka、Flume或者HDFS的流数据生成也可以由别的DStream经过各种转换操作得来。讲到这里你是不是觉得内容似曾相识

没错底层DStream也是由很多个序列化的RDD构成按时间片比如一秒切分成的每个数据单位都是一个RDD。然后Spark核心引擎将对DStream的Transformation操作变为针对Spark中对 RDD的Transformation操作将RDD经过操作变成中间结果保存在内存中。

之前的DataFrame和DataSet也是同样基于RDD所以说RDD是Spark最基本的数据抽象。就像Java里的基本数据类型Primitive Type一样所有的数据都可以用基本数据类型描述。

也正是因为这样无论是DataFrame还是DStream都具有RDD的不可变性、分区性和容错性等特质。

所以Spark是一个高度统一的平台所有的高级API都有相同的性质它们之间可以很容易地相互转化。Spark的野心就是用这一套工具统一所有数据处理的场景。

由于Spark Streaming将底层的细节封装起来了所以对于开发者来说只需要操作DStream就行。接下来让我们一起学习DStream的结构以及它支持的转换操作。

DStream

下图就是DStream的内部形式即一个连续的RDD序列每一个RDD代表一个时间窗口的输入数据流。

对DStream的转换操作意味着对它包含的每一个RDD进行同样的转换操作。比如下边的例子。

sc = SparkContext(master, appName)
ssc = StreamingContext(sc, 1)
lines = sc.socketTextStream("localhost", 9999)
words = lines.flatMap(lambda line: line.split(" "))

首先我们创建了一个lines的DStream去监听来自本机9999端口的数据流每一个数据代表一行文本。然后对lines进行flatMap的转换操作把每一个文本行拆分成词语。

本质上对一个DStream进行flatMap操作就是对它里边的每一个RDD进行flatMap操作生成了一系列新的RDD构成了一个新的代表词语的DStream。

正因为DStream和RDD的关系RDD支持的所有转换操作DStream都支持比如map、flatMap、filter、union等。这些操作我们在前边学习RDD时都详细介绍过在此不做赘述。

此外DStream还有一些特有操作如滑动窗口操作我们可以一起探讨。

滑动窗口操作

任何Spark Streaming的程序都要首先创建一个StreamingContext的对象它是所有Streaming操作的入口。

比如我们可以通过StreamingContext来创建DStream。前边提到的例子中lines这个DStream就是由名为sc的StreamingContext创建的。

StreamingContext中最重要的参数是批处理的时间间隔,即把流数据细分成数据块的粒度。

这个时间间隔决定了流处理的延迟性所以需要我们根据需求和资源来权衡间隔的长度。上边的例子中我们把输入的数据流以秒为单位划分每一秒的数据会生成一个RDD进行运算。

有些场景中我们需要每隔一段时间统计过去某个时间段内的数据。比如对热点搜索词语进行统计每隔10秒钟输出过去60秒内排名前十位的热点词。这是流处理的一个基本应用场景很多流处理框架如Apache Flink都有原生的支持。所以Spark也同样支持滑动窗口操作。

从统计热点词这个例子,你可以看出滑动窗口操作有两个基本参数:

  • 窗口长度window length每次统计的数据的时间跨度在例子中是60秒
  • 滑动间隔sliding interval每次统计的时间间隔在例子中是10秒。

显然由于Spark Streaming流处理的最小时间单位就是StreamingContext的时间间隔所以这两个参数一定是它的整数倍。

最基本的滑动窗口操作是window它可以返回一个新的DStream这个DStream中每个RDD代表一段时间窗口内的数据如下例所示。

windowed_words = words.window(60, 10)

windowed_words代表的就是热词统计例子中我们所需的DStream即它里边每一个数据块都包含过去60秒内的词语而且这样的块每10秒钟就会生成一个。

此外Spark Streaming还支持一些“进阶”窗口操作。如countByWindow、reduceByWindow、reduceByKeyAndWindow和countByValueAndWindow在此不做深入讨论。

Spark Streaming的优缺点

讲了这么多Spark Streaming不管内部实现也好支持的API也好我们还并不明白它的优势是什么相比起其他流处理框架的缺点是什么。只有明白了这些才能帮助我们在实际工作中决定是否使用Spark Streaming。

首先Spark Streaming的优点很明显由于它的底层是基于RDD实现的所以RDD的优良特性在它这里都有体现。

比如数据容错性如果RDD 的某些分区丢失了,可以通过依赖信息重新计算恢复。

再比如运行速度DStream同样也能通过persist()方法将数据流存放在内存中。这样做的好处是遇到需要多次迭代计算的程序时,速度优势十分明显。

而且Spark Streaming是Spark生态的一部分。所以它可以和Spark的核心引擎、Spark SQL、MLlib等无缝衔接。换句话说对实时处理出来的中间数据我们可以立即在程序中无缝进行批处理、交互式查询等操作。这个特点大大增强了Spark Streaming的优势和功能使得基于Spark Streaming的应用程序很容易扩展。

而Spark Streaming的主要缺点是实时计算延迟较高一般在秒的级别。这是由于Spark Streaming不支持太小的批处理的时间间隔。

在第二章中我们讲过准实时和实时系统无疑Spark Streaming是一个准实时系统。别的流处理框架如Storm的延迟性就好很多可以做到毫秒级。

小结

Spark Streaming作为Spark中的流处理组件把连续的流数据按时间间隔划分为一个个数据块然后对每个数据块分别进行批处理。

在内部每个数据块就是一个RDD所以Spark Streaming有RDD的所有优点处理速度快数据容错性好支持高度并行计算。

但是,它的实时延迟相比起别的流处理框架比较高。在实际工作中,我们还是要具体情况具体分析,选择正确的处理框架。

思考题

如果想要优化一个Spark Streaming程序你会从哪些角度入手

欢迎你把答案写在留言区,与我和其他同学一起讨论。

如果你觉得有所收获,也欢迎把文章分享给你的朋友。