# 30 | 流计算与消息(二):在流计算中使用Kafka链接计算任务 你好,我是李玥。 上节课我们一起实现了一个流计算的例子,并通过这个例子学习了流计算的实现原理。我们知道,流计算框架本身是个分布式系统,一般由多个节点组成一个集群。我们的计算任务在计算集群中运行的时候,会被拆分成多个子任务,这些子任务也是分布在集群的多个计算节点上的。 大部分流计算平台都会采用存储计算分离的设计,将计算任务的状态保存在HDFS等分布式存储系统中。每个子任务将状态分离出去之后,就变成了无状态的节点,如果某一个计算节点发生宕机,使用集群中任意一个节点都可以替代故障节点。 但是,对流计算来说,这里面还有一个问题没解决,就是在集群中流动的数据并没有被持久化,所以它们就有可能由于节点故障而丢失,怎么解决这个问题呢?办法也比较简单粗暴,就是直接重启整个计算任务,并且从数据源头向前回溯一些数据。计算任务重启之后,会重新分配计算节点,顺便就完成了故障迁移。 回溯数据源,可以保证数据不丢失,这和消息队列中,通过重发未成功的消息来保证数据不丢的方法是类似的。所以,它们面临同样的问题:可能会出现重复的消息。消息队列可以通过在消费端做幂等来克服这个问题,但是对于流计算任务来说,这个问题就很棘手了。 对于接收计算结果的下游系统,它可能会收到重复的计算结果,这还不是最糟糕的。像一些统计类的计算任务,就会有比较大的影响,比如上节课中统计访问次数的例子,本来这个IP地址在统计周期内被访问了5次,产生了5条访问日志,正确的结果应该是5次。如果日志被重复统计,那结果就会多于5次,重复的数据导致统计结果出现了错误。怎么解决这个问题呢? 我们之前提到过,Kafka支持Exactly Once语义,它的这个特性就是为了解决这个问题而生的。这节课,我们就来通过一个例子学习一下,如何使用Kafka配合Flink,解决数据重复的问题,实现端到端的Exactly Once语义。 ## Flink是如何保证Exactly Once语义的? 我们所说的端到端Exactly Once,这里面的“端到端”指的是,数据从Kafka的A主题消费,发送给Flink的计算集群进行计算,计算结果再发给Kafka的B主题。在这整个过程中,无论是Kafka集群的节点还是Flink集群的节点发生故障,都不会影响计算结果,每条消息只会被计算一次,不能多也不能少。 在理解端到端Exactly Once的实现原理之前,我们需要先了解一下,Flink集群本身是如何保证Exactly Once语义的。为什么Flink也需要保证Exactly Once呢?Flink集群本身也是一个分布式系统,它首先需要保证数据在Flink集群内部只被计算一次,只有在这个基础上,才谈得到端到端的Exactly Once。 Flink通过CheckPoint机制来定期保存计算任务的快照,这个快照中主要包含两个重要的数据: 1. 整个计算任务的状态。这个状态主要是计算任务中,每个子任务在计算过程中需要保存的临时状态数据。比如,上节课例子中汇总了一半的数据。 2. 数据源的位置信息。这个信息记录了在数据源的这个流中已经计算了哪些数据。如果数据源是Kafka的主题,这个位置信息就是Kafka主题中的消费位置。 有了CheckPoint,当计算任务失败重启的时候,可以从最近的一个CheckPoint恢复计算任务。具体的做法是,每个子任务先从CheckPoint中读取并恢复自己的状态,然后整个计算任务从CheckPoint中记录的数据源位置开始消费数据,只要这个恢复位置和CheckPoint中每个子任务的状态是完全对应的,或者说,每个子任务的状态恰好是:“刚刚处理完恢复位置之前的那条数据,还没有开始处理恢复位置对应的这条数据”,这个时刻保存的状态,就可以做到严丝合缝地恢复计算任务,每一条数据既不会丢失也不会重复。 因为每个子任务分布在不同的节点上,并且数据是一直在子任务中流动的,所以确保CheckPoint中记录的恢复位置和每个子任务的状态完全对应并不是一件容易的事儿,Flink是怎么实现的呢? Flink通过在数据流中插入一个Barrier(屏障)来确保CheckPoint中的位置和状态完全对应。下面这张图来自[Flink官网的说明文档](https://ci.apache.org/projects/flink/flink-docs-stable/internals/stream_checkpointing.html)。 ![](https://static001.geekbang.org/resource/image/0c/fa/0c301d798341dc53515611c31e9031fa.png) 你可以把Barrier理解为一条特殊的数据。Barrier由Flink生成,并在数据进入计算集群时被插入到数据流中。这样,无限的数据流就被很多的Barrier分隔成很多段。Barrier在流经每个计算节点的时候,就会触发这个节点在CheckPoint中保存本节点的状态,如果这个节点是数据源节点,还会保存数据源的位置。 当一个Barrier流过所有计算节点,流出计算集群后,一个CheckPoint也就保存完成了。由于每个节点都是在Barrier流过的时候保存的状态,这时的状态恰好就是Barrier所在位置(也就是CheckPoint数据源位置)对应的状态,这样就完美解决了状态与恢复位置对应的问题。 Flink通过CheckPoint机制实现了集群内计算任务的Exactly Once语义,但是仍然实现不了在输入和输出两端数据不丢不重。比如,Flink在把一条计算结果发给Kafka并收到来自Kafka的“发送成功”响应之后,才会继续处理下一条数据。如果这个时候重启计算任务,Flink集群内的数据都可以完美地恢复到上一个CheckPoint,但是已经发给Kafka的消息却没办法撤回,还是会出现数据重复的问题。 所以,我们需要配合Kafka的Exactly Once机制,才能实现端到端的Exactly Once。 ## Kafka如何配合Flink实现端到端Exactly Once? Kafka的Exactly Once语义是通过它的事务和生产幂等两个特性来共同实现的。其中Kafka事务的实现原理,我们在《[25 | RocketMQ与Kafka中如何实现事务?](https://time.geekbang.org/column/article/138724)》这节课中讲过。它可以保证一个事务内的所有消息,要么都成功投递,要么都不投递。 生产幂等这个特性可以保证,在生产者给Kafka Broker发送消息这个过程中,消息不会重复发送。这个实现原理和我们在《[05 | 如何确保消息不会丢失?](https://time.geekbang.org/column/article/111488)》这节课中介绍的“检测消息丢失”的方法是类似的,都是通过连续递增的序号进行检测。Kafka的生产者给每个消息增加都附加一个连续递增的序号,Broker端会检测这个序号的连续性,如果序号重复了,Broker会拒绝这个重复消息。 Kafka的这两个机制,配合Flink就可以来实现端到端的Exactly Once了。简单地说就是,每个Flink的CheckPoint对应一个Kafka事务。Flink在创建一个CheckPoint的时候,同时开启一个Kafka的事务,完成CheckPoint同时提交Kafka的事务。当计算任务重启的时候,在Flink中计算任务会恢复到上一个CheckPoint,这个CheckPoint正好对应Kafka上一个成功提交的事务。未完成的CheckPoint和未提交的事务中的消息都会被丢弃,这样就实现了端到端的Exactly Once。 但是,怎么才能保证“完成CheckPoint同时提交Kafka的事务”呢?或者说,如何来保证“完成CheckPoint”和“提交Kafka事务”这两个操作,要么都成功,要么都失败呢?这不就是一个典型的分布式事务问题嘛! 所以,Flink基于两阶段提交这个常用的分布式事务算法,实现了一分布式事务的控制器来解决这个问题。如果你对具体的实现原理感兴趣,可以看一下Flink官网文档中的[这篇文章](https://flink.apache.org/features/2018/03/01/end-to-end-exactly-once-apache-flink.html)。 ## Exactly Once版本的Web请求的统计 下面进入实战环节,我们来把上节课的“统计Web请求的次数”的Flink Job改造一下,让这个Job具备Exactly Once特性。这个实时统计任务接收NGINX的access.log,每5秒钟按照IP地址统计Web请求的次数。假设我们已经有一个实时发送access.log的日志服务来发送日志,日志的内容只包含访问时间和IP地址,这个日志服务就是我们流计算任务的数据源。 改造之后,我们需要把数据的来源替换成Kafka的ip\_count\_source主题,计算结果也要保存到Kafka的主题ip\_count\_sink中。 整个系统的数据流向就变成下图这样: ![](https://static001.geekbang.org/resource/image/b6/6d/b62a67148c0600a1814f763a70b2056d.jpg) 日志服务将日志数据发送到Kafka的主题ip\_count\_source,计算任务消费这个主题的数据作为数据源,计算结果会被写入到Kafka的主题ip\_count\_sink中。 Flink提供了Kafka Connector模块,可以作为数据源从Kafka中消费数据,也可以作为Kafka的Producer,将计算结果发送给Kafka,并且,这个Kafka Connector已经实现了Exactly Once语义,我们在使用的时候只要做适当的配置就可以了。 这次我们用Java语言来实现这个任务,改造后的计算任务代码如下: ``` public class ExactlyOnceIpCount { public static void main(String[] args) throws Exception { // 设置输入和输出 FlinkKafkaConsumer011 sourceConsumer = setupSource(); FlinkKafkaProducer011 sinkProducer = setupSink(); // 设置运行时环境 final StreamExecutionEnvironment env = StreamExecutionEnvironment.getExecutionEnvironment(); env.setStreamTimeCharacteristic(TimeCharacteristic.EventTime); // 按照EventTime来统计 env.enableCheckpointing(5000); // 每5秒保存一次CheckPoint // 设置CheckPoint CheckpointConfig config = env.getCheckpointConfig(); config.setCheckpointingMode(CheckpointingMode.EXACTLY_ONCE); // 设置CheckPoint模式为EXACTLY_ONCE config.enableExternalizedCheckpoints( CheckpointConfig.ExternalizedCheckpointCleanup.RETAIN_ON_CANCELLATION); // 取消任务时保留CheckPoint config.setPreferCheckpointForRecovery(true); // 启动时从CheckPoint恢复任务 // 设置CheckPoint的StateBackend,在这里CheckPoint保存在本地临时目录中。 // 只适合单节点做实验,在生产环境应该使用分布式文件系统,例如HDFS。 File tmpDirFile = new File(System.getProperty("java.io.tmpdir")); env.setStateBackend((StateBackend) new FsStateBackend(tmpDirFile.toURI().toURL().toString())); // 设置故障恢复策略:任务失败的时候自动每隔10秒重启,一共尝试重启3次 env.setRestartStrategy(RestartStrategies.fixedDelayRestart( 3, // number of restart attempts 10000 // delay )); // 定义输入:从Kafka中获取数据 DataStream input = env .addSource(sourceConsumer); // 计算:每5秒钟按照ip对count求和 DataStream output = input .keyBy(IpAndCount::getIp) // 按照ip地址统计 .window(TumblingEventTimeWindows.of(Time.seconds(5))) // 每5秒钟统计一次 .allowedLateness(Time.seconds(5)) .sum("count"); // 对count字段求和 // 输出到kafka topic output.map(IpAndCount::toString).addSink(sinkProducer); // execute program env.execute("Exactly-once IpCount"); } } ``` 这段代码和上节课中原始版本的代码整体架构是差不多的,同样是:定义数据源、定义计算逻辑和定义输入这三大步骤。下面主要来说不同之处,这些不同的地方也就是如何配置Exactly Once特性的关键点。 首先,我们需要开启并配置好CheckPoint。在这段代码中,我们开启了CheckPoint,设置每5秒钟创建一个CheckPoint。然后,还需要定义保存CheckPoint的StateBackend,也就是告诉Flink把CheckPoint保存在哪儿。在生产环境中,CheckPoint应该保存到HDFS这样的分布式文件系统中。我们这个例子中,为了方便运行调试,直接把CheckPoint保存到本地的临时目录中。之后,我们还需要将Job配置成自动重启,这样当节点发生故障时,Flink会自动重启Job并从最近一次CheckPoint开始恢复。 我们在定义输出创建FlinkKafkaProducer的时候,需要指定Exactly Once语义,这样Flink才会开启Kafka的事务,代码如下: ``` private static FlinkKafkaProducer011 setupSink() { // 设置Kafka Producer属性 Properties producerProperties = new Properties(); producerProperties.put("bootstrap.servers", "localhost:9092"); // 事务超时时间设置为1分钟 producerProperties.put("transaction.timeout.ms", "60000"); // 创建 FlinkKafkaProducer,指定语义为EXACTLY_ONCE return new FlinkKafkaProducer011<>( "ip_count_sink", new KeyedSerializationSchemaWrapper<>(new SimpleStringSchema()), producerProperties, FlinkKafkaProducer011.Semantic.EXACTLY_ONCE); } ``` 最后一点需要注意的,在从Kafka主题ip\_count\_sink中消费计算结果的时候,需要配置Consumer属性:isolation.level=read\_committed,也就是只消费已提交事务的消息。因为默认情况下,Kafka的Consumer是可以消费到未提交事务的消息的。 这个例子的完整代码我放到了GitHub上,编译和运行这个例子的方法我也写在了项目的README中,你可以点击[这里](https://github.com/liyue2008/kafka-flink-exactlyonce-example)查看。 ## 小结 端到端Exactly Once语义,可以保证在分布式系统中,每条数据不多不少只被处理一次。在流计算中,因为数据重复会导致计算结果错误,所以Exactly Once在流计算场景中尤其重要。Kafka和Flink都提供了保证Exactly Once的特性,配合使用可以实现端到端的Exactly Once语义。 在Flink中,如果节点出现故障,可以自动重启计算任务,重新分配计算节点来保证系统的可用性。配合CheckPoint机制,可以保证重启后任务的状态恢复到最后一次CheckPoint,然后从CheckPoint中记录的恢复位置继续读取数据进行计算。Flink通过一个巧妙的Barrier使CheckPoint中恢复位置和各节点状态完全对应。 Kafka的Exactly Once语义是通过它的事务和生产幂等两个特性来共同实现的。在配合Flink的时候,每个Flink的CheckPoint对应一个Kafka事务,只要保证CheckPoint和Kafka事务同步提交就可以实现端到端的Exactly Once,Flink通过“二阶段提交”这个分布式事务的经典算法来保证CheckPoint和Kafka事务状态的一致性。 可以看到,Flink配合Kafka来实现端到端的Exactly Once语义,整个实现过程比较复杂,但是,这个复杂的大问题是由一个一个小问题组成的,每个小问题的原理都是很简单的。比如:Kafka如何实现的生产幂等?Flink如何通过存储计算分离解决子任务状态恢复的?很多这些小问题和我们课程中遇到的类似问题是差不多的,那你就可以用到我们学习过的解决方法。 你需要重点掌握的是,每一个小问题它面临的场景是什么样的,以及如何解决问题的方法。而不要拘泥于,Kafka或者Flink的某个参数怎么配这些细节问题。这些问题可以等到你在生产中真正需要使用的时候,再去读文档,“现学现卖”都来得及。 ## 思考题 我们的课程中反复强调过,在消息队列的消费端,一定要“先执行消费业务逻辑,再确认消费”,这样才能保证不丢数据。我们这节课中,并没有提到FlinkKafkaConsumer在从数据源主题ip\_count\_sink消费数据之后,如何来确认消费的。如果消费位置管理不好,一样会导致消息丢失或者重复,课后请你查看一下相关的文档和源代码,看一下FlinkKafkaConsumer是如何来确认消费的。欢迎在留言区与我分享讨论。 感谢阅读,如果你觉得这篇文章对你有一些启发,也欢迎把它分享给你的朋友。