Flink学习(十八) 状态管理与状态编程

Flink中的状态

 

 

由一个任务维护,并且用来计算某个结果的所有数据,都属于这个任务的状态;可以认为状态就是一个本地变量,可以被任务的业务逻辑访问;Flink会进行状态管理,包括状态一致性,故障处理以及高效存储和访问,以使开发人员可以专注于应用程序的逻辑。

在Flink中,状态始终与特定算子相关联,为了使运行的Flink了解算子的状态,算子需要预先注册其状态。

总的来说,有两种类型的状态:

算子状态(Operator State): 算子状态的作用范围限定为算子任务。

键控状态(Keyed State): 根据输入数据流中定义的键(key)来维护和访问。

 

算子状态(Operator State)

 

 

算子状态的作用范围限定为算子任务,由同一个并行任务所处理的所有数据都可以访问到相同的状态,状态对于同一任务而言使共享的,算子状态不能由相同或不同算子的另一个任务访问。

 

算子状态数据结构

列表状态(List State): 将状态表示为一组数据的列表。

联合列表状态(Union list state): 也将状态表示为数据的列表,它与常规列表状态的区别在于,在发生故障时,或者从保存点(savepoint)启动应用程序时如何恢复。

广播状态(Broadcast state): 如果一个算子有多项任务,而它的每项任务状态又都相同,那么这种特殊情况最适合应用广播状态。

 

键控状态(Keyed State)

 

 

键控状态时根据输入数据流中定义的键(key)来维护和访问的,Flink为每隔key维护一个状态实例,并将具有相同键的所有数据,都分区到同一个算子任务中,这个任务会维护和处理这个key对应的状态,当文物处理一条数据时,它会自动将状态的访问范围限定为当前数据的key。

 

键控状态数据结构

值状态(Value state): 将状态表示为单个的值。

列表状态(list state): 将状态表示为一组数据的列表。

映射状态(Map state): 将状态表示为一组key-value对。

聚合状态(Reducing state & aggregating State): 将状态表示为一个用于聚合操作的列表。

 

键控状态的使用(值状态(Value state)举例:)

声明一个键控状态

//定义一个状态,用来保存上一个数据的温度值
//用懒加载的方式,一开始定义的时候我们还不执行,等到调用的时候去执行
//所有的状态都这么定义 当成一个变量直接用
lazy val lastTemp: ValueState[Double] = getRuntimeContext.getState(new ValueStateDescriptor[Double]("lastTemp", classOf[Double]))

读取状态

//每来一条数据的时候,从状态中取出上一次的温度值

val preTemp = lastTemp.value()

对状态赋值

//更新温度值

 lastTemp.update(value.temperature)

 

状态后端(State Backends)

每传入一条数据,有状态的算子任务都会读取和更新状态。由于有效的状态访问对于处理数据的低延迟至关重要,因此每隔并行任务都会在本地维护其状态,以确保快速的状态访问。状态的存储,访问以及维护,由一个可插入的组件决定,这个组件就叫做状态呀

后端(state backend),状态后端主要负责两件事:一是本地的状态管理,以及将检查点(checkpoint)状态写入远程存储。

    //开启checkpoint 传入参数代表每隔多久进行checkpoint
    env.enableCheckpointing(60000)

    //状态后端
    //    env.setStateBackend(new MemoryStateBackend())
    //    env.setStateBackend(new RocksDBStateBackend(""))

 

 

 

 

状态编程小应用(三种方式实现)

/**
* 需求:
* 检测两次温度超过一定的范围的话 就报警
* 传感器的温度 跳变太大
*/

方式一:继承 KeyedProcessFunction 抽象类
class TempChangeAlert(thresHold: Double) extends KeyedProcessFunction[String, SensorReading, (String, Double, Double)] {
  //定义一个状态变量,保存上次的温度值
  lazy val lastTempState: ValueState[Double] = getRuntimeContext.getState(new ValueStateDescriptor[Double]("lastTemp", classOf[Double]))


  override def processElement(value: SensorReading, ctx: KeyedProcessFunction[String, SensorReading, (String, Double, Double)]#Context, out: Collector[(String, Double, Double)]): Unit = {

    //获取上一次的温度值
    val lastTemp = lastTempState.value()

    //用当前的温度值和上次的温度值计算一个差值,如果大于预设值,就报警‘
    val diff = (value.temperature - lastTemp).abs
    if(lastTemp==0.0){
      println(s"方法一,第一条数据进来,$value")
    }else if(diff > thresHold) {
      out.collect((value.id, lastTemp, value.temperature))
    }

    //做完一次处理之后将状态进行更新
    lastTempState.update(value.temperature)
  }
}

 

方式二:继承 RichFlatMapFunction 抽象类
class TempChangeAlertFlatMap(thresHold: Double) extends RichFlatMapFunction[SensorReading,(String,Double,Double)]{


  //定义
  private var lastTempState:ValueState[Double] = _


  override def open(parameters: Configuration): Unit = {
    //初始化的时候声明变量
    lastTempState = getRuntimeContext.getState(new ValueStateDescriptor[Double]("lastTemp",classOf[Double]))
  }

  override def flatMap(value: SensorReading, out: Collector[(String, Double, Double)]): Unit = {

    //获取上一次的温度值
    val lastTemp = lastTempState.value()

    //用当前的温度值和上次的温度值计算一个差值,如果大于预设值,就报警‘
    val diff = (value.temperature - lastTemp).abs
    if(lastTemp==0.0){
      println(s"方法二,第一条数据进来,$value")
    }else if(diff > thresHold) {
      out.collect((value.id, lastTemp, value.temperature))
    }

    //做完一次处理之后将状态进行更新
    lastTempState.update(value.temperature)
  }
}

 

方式三:keyBy后调用 flatMapWithState 方法

 //实现方式3
    val processedStream4 = dataStream.keyBy(_.id)
        .flatMapWithState[(String,Double,Double),Double]{
      //没有状态的话,也就是没有数据来过,那么就将当前数据温度值存入状态
      case (input:SensorReading,None) => {
        println(s"方法三,第一条数据进来,$input")

        (List.empty,Some(input.temperature))
      }
        //如果有状态的话,就与上一次的温度值比较差值,如果大于预设值,就报警
      case (input:SensorReading,lastTemp:Some[Double]) => {
        val diff = (input.temperature - lastTemp.get).abs
        if(diff>10.0){
          (List((input.id,lastTemp.get,input.temperature)),Some(input.temperature))
        }else{
          (List.empty,Some(input.temperature))
        }
      }
    }

 

完整代码:

package com.wyh.statebackendApi


import org.apache.flink.api.common.functions.RichFlatMapFunction
import org.apache.flink.api.common.state.{ValueState, ValueStateDescriptor}
import org.apache.flink.configuration.Configuration
import org.apache.flink.streaming.api.TimeCharacteristic
import org.apache.flink.streaming.api.functions.KeyedProcessFunction
import org.apache.flink.streaming.api.functions.timestamps.BoundedOutOfOrdernessTimestampExtractor
import org.apache.flink.streaming.api.scala._
import org.apache.flink.streaming.api.windowing.time.Time
import org.apache.flink.util.Collector


/**
  * 需求:
  * 检测两次温度超过一定的范围的话 就报警
  * 传感器的温度 跳变太大
  */

//温度传感器读数样例类
case class SensorReading(id: String, timestamp: Long, temperature: Double)

object Demo1 {
  def main(args: Array[String]): Unit = {
    val env = StreamExecutionEnvironment.getExecutionEnvironment

    env.setParallelism(1)
    env.setStreamTimeCharacteristic(TimeCharacteristic.EventTime)

    //开启checkpoint 传入参数代表每隔多久进行checkpoint
    env.enableCheckpointing(60000)

    //状态后端
    //    env.setStateBackend(new MemoryStateBackend())
    //    env.setStateBackend(new RocksDBStateBackend(""))


    val stream = env.socketTextStream("localhost", 7777)

    //Transform操作
    val dataStream: DataStream[SensorReading] = stream.map(data => {
      val dataArray = data.split(",")
      SensorReading(dataArray(0).trim, dataArray(1).trim.toLong, dataArray(2).trim.toDouble)
    })
      //===到来的数据是升序的,准时发车,用assignAscendingTimestamps
      //指定哪个字段是时间戳 需要的是毫秒 * 1000
      //      .assignAscendingTimestamps(_.timestamp * 1000)
      //===处理乱序数据
      //      .assignTimestampsAndWatermarks(new MyAssignerPeriodic())
      //==底层也是周期性生成的一个方法 处理乱序数据 延迟1秒种生成水位 同时分配水位和时间戳 括号里传的是等待延迟的时间
      .assignTimestampsAndWatermarks(new BoundedOutOfOrdernessTimestampExtractor[SensorReading](Time.seconds(1)) {
      override def extractTimestamp(t: SensorReading): Long = {
        t.timestamp * 1000
      }
    })

//    val processedStream = dataStream.keyBy(_.id)
//      .process(new TempIncreAlert())

    //实现方式1
    val processedStream2 = dataStream.keyBy(_.id)
      .process(new TempChangeAlert(10.0))


    //实现方式2
    val processedStream3 = dataStream.keyBy(_.id)
        .flatMap(new TempChangeAlertFlatMap(10.0))

    //实现方式3
    val processedStream4 = dataStream.keyBy(_.id)
        .flatMapWithState[(String,Double,Double),Double]{
      //没有状态的话,也就是没有数据来过,那么就将当前数据温度值存入状态
      case (input:SensorReading,None) => {
        println(s"方法三,第一条数据进来,$input")

        (List.empty,Some(input.temperature))
      }
        //如果有状态的话,就与上一次的温度值比较差值,如果大于预设值,就报警
      case (input:SensorReading,lastTemp:Some[Double]) => {
        val diff = (input.temperature - lastTemp.get).abs
        if(diff>10.0){
          (List((input.id,lastTemp.get,input.temperature)),Some(input.temperature))
        }else{
          (List.empty,Some(input.temperature))
        }
      }
    }

    processedStream2.printToErr("processedStream2 data")

    processedStream3.printToErr("processedStream3 data")

    processedStream4.printToErr("processedStream4 data")

    dataStream.print("input data")


    env.execute("StateBackend Test")

  }

}

class TempIncreAlert() extends KeyedProcessFunction[String, SensorReading, String] {
  //定义一个状态,用来保存上一个数据的温度值
  //用懒加载的方式,一开始定义的时候我们还不执行,等到调用的时候去执行
  //所有的状态都这么定义 当成一个变量直接用
  lazy val lastTemp: ValueState[Double] = getRuntimeContext.getState(new ValueStateDescriptor[Double]("lastTemp", classOf[Double]))
  //定义一个状态用来保存定时器的时间戳
  lazy val currentTimer: ValueState[Long] = getRuntimeContext.getState(new ValueStateDescriptor[Long]("currentTimer", classOf[Long]))

  //判断温度连续上升
  //跟上一次数据进行比较 如果比较一直大 10秒种内进行报警
  //注册一个定时器 把上一次的数据保存成当前的状态
  override def processElement(value: SensorReading, ctx: KeyedProcessFunction[String, SensorReading, String]#Context, out: Collector[String]): Unit = {
    //每来一条数据的时候,从状态中取出上一次的温度值
    val preTemp = lastTemp.value()
    var curTimerTs = currentTimer.value()

    //更新温度值
    lastTemp.update(value.temperature)

    //加个if判断最开始的温度是否为0来判断是否是第一条数据 温度上升且没有设置过定时器,则注册定时器
    if (preTemp == 0.0) {
      println("这是第一条数据进来")
    } else if ((value.temperature > preTemp) && (curTimerTs == 0L)) {

      val timerTs = ctx.timerService().currentProcessingTime() + 10000L

      //传入当前时间加1  是时间戳
      ctx.timerService().registerProcessingTimeTimer(timerTs)
      currentTimer.update(timerTs)
    } else if (value.temperature <= preTemp) {
      //如果温度下降 或者是第一条数据 删除定时器
      ctx.timerService().deleteProcessingTimeTimer(curTimerTs)
      //删除定时器之后将状态清空
      currentTimer.clear()
    }
  }

  //在回调函数中执行定时器到的逻辑
  //当前的时间 ctx上下文 out输出信息
  override def onTimer(timestamp: Long, ctx: KeyedProcessFunction[String, SensorReading, String]#OnTimerContext, out: Collector[String]): Unit = {
    //直接输出报警信息
    out.collect(ctx.getCurrentKey + "温度连续上升")
    //考虑真实情况,将状态都清空
    currentTimer.clear()
  }
}

class TempChangeAlert(thresHold: Double) extends KeyedProcessFunction[String, SensorReading, (String, Double, Double)] {
  //定义一个状态变量,保存上次的温度值
  lazy val lastTempState: ValueState[Double] = getRuntimeContext.getState(new ValueStateDescriptor[Double]("lastTemp", classOf[Double]))


  override def processElement(value: SensorReading, ctx: KeyedProcessFunction[String, SensorReading, (String, Double, Double)]#Context, out: Collector[(String, Double, Double)]): Unit = {

    //获取上一次的温度值
    val lastTemp = lastTempState.value()

    //用当前的温度值和上次的温度值计算一个差值,如果大于预设值,就报警‘
    val diff = (value.temperature - lastTemp).abs
    if(lastTemp==0.0){
      println(s"方法一,第一条数据进来,$value")
    }else if(diff > thresHold) {
      out.collect((value.id, lastTemp, value.temperature))
    }

    //做完一次处理之后将状态进行更新
    lastTempState.update(value.temperature)
  }
}


class TempChangeAlertFlatMap(thresHold: Double) extends RichFlatMapFunction[SensorReading,(String,Double,Double)]{


  //定义
  private var lastTempState:ValueState[Double] = _


  override def open(parameters: Configuration): Unit = {
    //初始化的时候声明变量
    lastTempState = getRuntimeContext.getState(new ValueStateDescriptor[Double]("lastTemp",classOf[Double]))
  }

  override def flatMap(value: SensorReading, out: Collector[(String, Double, Double)]): Unit = {

    //获取上一次的温度值
    val lastTemp = lastTempState.value()

    //用当前的温度值和上次的温度值计算一个差值,如果大于预设值,就报警‘
    val diff = (value.temperature - lastTemp).abs
    if(lastTemp==0.0){
      println(s"方法二,第一条数据进来,$value")
    }else if(diff > thresHold) {
      out.collect((value.id, lastTemp, value.temperature))
    }

    //做完一次处理之后将状态进行更新
    lastTempState.update(value.temperature)
  }
}

 

运行测试:

输入数据:

 

 

结果:

posted @ 2020-05-30 21:36  Xiaohu_BigData  阅读(1508)  评论(0编辑  收藏  举报