Flink教程(21) EventTime事件时间 Watermark水位线 注意点

一、前言

Flink中窗口、事件时间、水位线都是很重要的概念,网上介绍他们的博客非常多。

我就没有必要像写八股文那样再写一遍。

窗口、事件时间、水位线这3个东西放在一起,他们其实解决一个问题,那就是

窗口在什么时候关闭,并触发计算?

下面就扯一扯关于这个问题的注意点

二、事件时间

1. 时间一定是精确到毫秒

在数据流中,表示事件时间的格式多种多样。

可能是精确到秒的yyyy-MM-dd HH:mm:ss或者长整型如1611076383。

但是Flink中的时间一定到精确到毫秒

在分配时间戳和水位线调用assignTimestampsAndWatermarks的时候,一定要注意extractTimestamp的返回值是long,是要精确到毫秒

2. 注意时区问题,我们在东八区

时间戳是从1970年1月1日0时0分0秒开始的,但是它是指的伦敦时间。我们中国常用的时间是东八区时间。

就是我们这边的时间和伦敦时间差8个小时。

我们获取数据流中的时间通常是东八区的时间,但如果Flink划分窗口是按照一天划分窗口,按照伦敦时间划分,那么这个时间窗口是有问题的。

所以对于这个时间在读取数据源时,对时间处理下,或者Flink有这个设置时区的功能话就设置下。

3. 时间窗口是左闭右开

时间窗口是有开始时间、结束时间的。

例如一个窗口是10:00:00 到10:10:00,窗口长度是10分钟。

但是事件时间是10:10:00整的数据,不属于这个窗口。

本窗口的最大时间戳 = 结束时间 - 1ms,窗口是左闭右开的。

4. 时间窗口的开始时间如何计算?

时间窗口的开始时间如何计算?

例如一个事件时间窗口长度是10分钟。

此时10:05:00的事件时间数据来了,Flink如何划分窗口呢?计算公式如下:

start = timestamp - (timestamp - offset + windowSize) %windowSize

offset偏移量默认为0。

(10:05 - 0 + 10分钟) %10分钟 = 5分钟,因为一个小时都是10分钟的整数倍

10:05 - 5分钟 = 10:00,说明窗口是从10:00开始,到10:10结束。

5. Flin1.12开始,默认时间特性是事件时间

Flin1.12开始,默认时间特性是事件时间

你会发现下面的语句提示已经过时了。

env.setStreamTimeCharacteristic(TimeCharacteristic.EventTime);

用新的时间窗口分配器来指定用的是事件时间还是处理时间

.window(TumblingProcessingTimeWindows.of(Time.seconds(5)))

.window(TumblingEventTimeWindows.of(Time.seconds(30)))

三、水位线

1. 水位线的特点

  • 水位线是用来解决数据乱序问题,本身是一个特殊的时间戳
  • 水位线单调递增的,Flink认为事件时间早于水位线的数据都来了

2. 水位线控制窗口何时结束

当前的watermark >= 窗口的最大时间戳时,窗口关闭。

在之前的博客Flink教程(7)时间特性、窗口、Watermark代码实践有水位线的图解。

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
要获取Flink中的EventTime,需要在数据源中定义一个时间戳(timestamp)和一个水印(watermark)。时间戳表示事件发生的时间,水印表示事件时间的上界。 在Flink中,可以通过实现AssignerWithPeriodicWatermarks接口来定义数据源的timestamp和watermark。具体步骤如下: 1. 创建一个类并实现AssignerWithPeriodicWatermarks接口。 2. 在该类中实现extractTimestamp方法,该方法返回一个long类型的时间戳。 3. 在该类中实现getCurrentWatermark方法,该方法返回一个Watermark类型的水印。 4. 在Flink程序中,使用assignTimestampsAndWatermarks方法将AssignerWithPeriodicWatermarks应用到数据源。 以下是一个简单的例子: ``` public class MyAssigner implements AssignerWithPeriodicWatermarks<MyEvent> { private final long maxOutOfOrderness = 3500; // 最大乱序时间为3.5秒 private long currentMaxTimestamp; @Override public long extractTimestamp(MyEvent event, long previousElementTimestamp) { long timestamp = event.getTimestamp(); currentMaxTimestamp = Math.max(timestamp, currentMaxTimestamp); return timestamp; } @Override public Watermark getCurrentWatermark() { return new Watermark(currentMaxTimestamp - maxOutOfOrderness); } } DataStream<MyEvent> stream = env .addSource(new MySource()) .assignTimestampsAndWatermarks(new MyAssigner()); ``` 在上面的例子中,MyEvent类必须包含一个getTimestamp方法来获取事件时间。MyAssigner类实现了AssignerWithPeriodicWatermarks接口,并在extractTimestamp方法中返回事件时间戳,在getCurrentWatermark方法中返回水印。最后,使用assignTimestampsAndWatermarks方法将MyAssigner应用到数据源上。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

瑟王

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值