1、为什么引入Backpressure
默认情况下,Spark Streaming
通过Receiver以生产者生产数据的速率接收数据,计算过程中会出现batch processing time > batch interval
的情况,其中batch processing time
为实际计算一个批次花费时间, batch interval
为Streaming应用设置的批处理间隔。这意味着Spark Streaming的数据接收速率高于Spark从队列中移除数据的速率,也就是数据处理能力低,在设置间隔内不能完全处理当前接收速率接收的数据。如果这种情况持续过长的时间,会造成数据在内存中堆积,导致Receiver所在Executor内存溢出等问题(如果设置StorageLevel包含disk, 则内存存放不下的数据会溢写至disk, 加大延迟)。Spark 1.5以前版本,用户如果要限制Receiver的数据接收速率,可以通过设置静态配制参数spark.streaming.receiver.maxRate
的值来实现,此举虽然可以通过限制接收速率,来适配当前的处理能力,防止内存溢出,但也会引入其它问题。比如:producer数据生产高于maxRate,当前集群处理能力也高于maxRate,这就会造成资源利用率下降等问题。为了更好的协调数据接收速率与资源处理能力,Spark Streaming 从v1.5开始引入反压机制(back-pressure)
,通过动态控制数据接收速率来适配集群数据处理能力。
2、Backpressure
Spark Streaming Backpressure: 根据JobScheduler反馈作业的执行信息来动态调整Receiver数据接收率。通过属性spark.streaming.backpressure.enabled
来控制是否启用backpressure机制,默认值false
,即不启用。
3、反压机制实现
SparkStreaming的开启反压机制其实就是给StreamingContext添加一个StreamingListener,在JobScheduler中的start代码如下:
具体可参考Spark Streaming