Kakfa官方设计文档解读

最近在研究kafka,原版论文见我的另一篇文章,本文对kafka2.7最新版的设计文档做一个概括性解读,原文见官网的设计文档。

1. Motivation

kafka被设计成一个通用的数据流处理平台。(早已超出2011年出版的日志收集器和消息队列)。

  • 高吞吐(日志聚合)
  • 大型数据块的整合(offline 平台的拉取)
  • 低延迟(消息队列)
  • 分区、分布式的实时流处理,从一个流派生新的流;
  • 容错(当被投喂非日志数据时)

为了支持这些特性,一系列组件被开发,使得kafka更像是一个DB的日志,而不是消息队列。

2. 持久性

不要害怕文件系统!
选择用文件系统,而不是把日志存进进程的缓存再统一flush,有以下理由:

  1. 顺序读写是随机读写的3000x(机械硬盘)
  2. OS disk page cache 收益很大,有时缓存进进程反而是double cache了
  3. JVM的堆内存管理比较鸡肋,GC开销也大
  4. 经过原子化的访问和高效压缩数据成字节序列,可以更高效使用内存
  • 为什么不用树结构?
    B-Tree的结构通用性很强,O(logN)的操作复杂度对于很多系统都足够了。但是,树结构本身不具备扩展性,尤其是其需要随机磁盘读写。在固定cache的情况下,树形结构的性能最好也是超线性的,随着数据量的增加。
    对于磁盘的顺序写是log的普遍选择。这里写不会阻塞读,写操作都是O(1)的复杂度。最重要的是,这里和数据量的大小不再有关系。因此,kafka可以将消息持久化保持7天,以供重复读。

3. 效率

由于消费是规模最大的操作,所以我们要尽可能把消费做的“轻”。

  • disk efficiency:类似系统问题主要在两方面:I/O操作太多+Bytes过度拷贝。
    之前已经讨论过disk方面的选择。I/O操作太多,kafka的解决方案是利用消息组的抽象概念,用大块的消息读写(生产/消费)来均摊网络代价。
    Bytes过度拷贝问题,kafka让producer, comsumer, broker采用同样的序列化协议,开辟了优化空间,然后利用Sendfile系统调用减少Copy。(具体见我的另一篇文章)

  • End-to-end Batch Compression
    保证传输效率的另一点就是压缩。kafka可以支持批压缩,主要是因为不同log之间经常会产生大量的重复。相比于端上的单条日志压缩,可以有更好的压缩比。压缩的消息会被写入磁盘,会被发送给Consumer,最终由消费者解压缩。支持的压缩方案有GZIP, Snappy, LZ4 and ZStandard等。

4. Producer

4.1 负载均衡

producer选好了partition,broker会直接answer给producer这个partition的leader所在的broker,然后直接传输,没有中间的路由层。
producer如何选partition呢?可以随机,也可以自选分区key和分区函数。以满足一些本地性。

4.2 异步发送

批量发送。可配置的定时/定量进行buffer batch send。

5. Consumer

5.1 Push vs. pull

  • push-fashion的系统,比如flume,难点在于对于多个消费者,没有办法根据消费者的接收能力控制消费速度。pull-fashion在这点要灵活的多。
  • 可以由consumer主动去进行批量拉取(用户配置),而不是靠broker猜测。
  • 如果broker暂时没数据,consumer不会忙等,会把自己阻塞掉,定期轮询。

5.2 Consumer Position

消息的消费状态由消费者保存(offset)。
如果由broker保存,会有一系列问题。吞吐量肯定会低,其次,如果消费者消费了,但是没有Ack,那么将来会重复消费,以及broker一系列tricky的问题。

5.3 Offline Data Load

由于kafka的持久存储系统,数据仓库/HDFS,会选择周期性的批量bulk load数据到数仓中,这就对kafka对大量数据的吞吐效率有保证。
对于此,kafka将数据负载平均分割,然后并行化,每一个split都是一个map task,也可以做combination。某个map task fail掉了也不要紧,可以直接从最初的位置重新开始。

5.4 Static Memebership

为了避免频繁触发rebalance导致Stop-The-World,kafka 2.3之后设置了静态成员,由用户设置其consumer-id,好处是当短暂离线退组,再上线进组时,使用同一个id,不会导致rebalance。但是其他broker和consumer变更的情况下,仍然必须要rebalance保证负载均衡。
(rebalance具体算法见我另一篇文章)

6. 消息传递语义

在kafka中,日志是有提交的概念的(具体的见下一节),如果日志提交了,只要复制了这个分区的broker有一个活着的,日志就还在。在本节我们假设broker本身不会丢失数据,以便理解对producer/consumer的消息传递保证。

6.1 Producer Delivery

对于producer来说,如果出现网络错误,是没法知道传输的日志是否已经提交了。在0.11.0版本以前,如果没有收到ack,那么没有别的办法,只能重传,这实际上就是至少一次的语义。
然而在0.11.0版本之后,kafka为每个消息提供了Sequence number,为每个producer分发id,这样broker的接收操作,可以设置为幂等的,就完成了对producer的确切一次的语义。
而且也是从0.11.0开始,producer对多个topic partitions发送数据也可以保证事务性,要么全部接收,要么全都没接收。

具体到使用的时候,producer可以根据消息类型自主选择持久化级别。log信息可以完全异步发送,当有重要数据时也可以选择有回调函数的Send,等待commit时block掉,commit的级别也可以设置,是leader收到即可或者需要多少个follower副本。一般来说,同步的Send在10ms这个级别。

6.2 Consumer Delivery

由于上面我们说,producer可以对多个topic partitions进行事务性的写(同时写成功或同时不成功)。这给kafka的一个场景带来了极大的便利:流处理。流处理就是通过一个topic经过一些变换产出到另一个topic中去,整个过程都在kafka集群中完成。我们把两条消息组成一个事务:转换后的消息+消费的offset。利用producer的事务写,要么offset和数据同时写入,要么同时没有被写入,这就达成了消费端的确切一次语义。
如果事务中途abort掉了,对于consumer有两种可见性,取决于consumer的隔离性级别

  1. read_uncommited:可以看到没提交的写
  2. read_commited:看不到没提交的写

上面说的是流处理的过程是可以达成确切一次语义。对于consumer来自外部系统呢?麻烦在于要把消费者的位置(broker知道)和实际消费的日志(consumer知道)同步起来,一个通用的做法是进行两阶段提交(编者注:kafka集群作为coordinator,每一个consumer作为一个worker)。然而很多外部系统(比如HDFS)并不支持两阶段提交。因此只能用一个更轻型也更通用的方案,让每一个consumer把自己的offset和实际数据放在同一个位置。有一点不妙的是,由于此时的消息没有主键,因此也无法进行去重(编者注:offset不可以作为消息的主键么?)。最终支持的是至少一次语义。

7. Replication

手工配置副本个数。副本个数为1就是不复制。
所有的读写都走leader,只在leader挂了,follower才用于自动故障转移。
kafka在复制容错方面,只考虑宕机/恢复模型,不会考虑分布式系统领域的拜占庭故障,即故意发错误信息的特殊情况。
kafka判定节点是否alive有两个条件:

  1. 是否和zookeeper的session心跳保持联系;
  2. 是否和leader落后在一定范围内(用户参数)。

对于一个partition, follower和leader共同构成副本集, follower像是consumer一样去拉取leader的日志。leader和alive follower共同构成ISR,leader时刻通过zookeeper跟踪ISR集合,剔除死掉的follower。

producer可以在持久性吞吐率之间做权衡。可以设置mininum replica must write。producer有几个选择:

  1. 完全不需要ack
  2. 需要ack,但只要leader的就可以
  3. 需要ack,要至少mininum副本写入(minimum ISR)

对于1.2.这两种选择,可能只有leader写了日志,然后就被消费了。
因此kafka的保障是:对于提交的消息,只要有一个副本活着,就不会丢失。

kafka对于节点短时间宕机恢复有容错保障,但是对网络分区就不再保证可用了。

7.1 Replicated Logs: Quorums, ISRs, and State Machines (Oh my!)

kafka采用的是replicated log模型,即消息由leader定序,Follower无脑copy即可。
如果leader宕机掉了,就要在ISR中启动多数选举。(Raft, Paxos等),最接近kafka的是MS的PacificA。
对于宕机恢复的节点,kafka不要求它的数据完全一致,但是在加入ISR之前,它的数据必须得到全量的恢复。

7.2 trade-off between availability and duribility

上面说过producer有三种选择,对于第三种选择,即最小ISR基数,存在一个trade-off。过于大的minimum容易导致分区不可用,必须阻塞等待有足够多的ISR;过于小的minimum容易导致数据丢失,比如minimum=1,那么实际上只有leader写入了。这里存在一个权衡,要用户把握。

7.3 Replica Management

kafka用round-rubin的方式保证某个topic的partitions不会聚集在少量的节点中。同样,也会用同样的方式保证leaders不会聚集在少量节点中。
另一方面,一般kafka由节点挂掉,是broker直接挂掉,不会是某个partition挂掉,那么一个broker挂掉,可能会触发几个甚至几十个partition的重新选举/rebalance。此时kafka的策略是选择另外一个broker,在更高的级别上领导这些partition的leader选举,这样使得选举过程可以批量化,更为高效。

8. Log Compaction

kafka log压缩保证在一个topic partition内,在消息内部每个key的最新值都会被保留下来。这意味着在任意时刻,我们能拿到当前各个key的最新快照。这在一些事务型的日志中非常重要,可以用于下游的数据恢复。
比如下图这三次更改中,只有最后一条记录不会被压缩。


image.png

这样的话其实就产生了两种保留策略,一种是默认的按照时间(7天)或者大小来保留;另一种是按照压缩来保留。

8.1 Log Compaction Basics

这是一个kafka log的逻辑视图。


image.png

真实的log compaction大概是这个样子的。offset即使被压缩也永远不会变,以免含义混淆。


image.png

对于log compaction,kafka给出了一些保证:
  • 消息会在一个可配置的时间之后才会进入log尾,可压缩;也就是说,如果一直在监听消费的consumer可以收到连续offset的消息,不会立即被压缩;
  • 消息的顺序不会被打乱,只是有些消息就被删除了;
  • 消息的offset不变;

(编者注:本节中有关消息删除的暂时略过)

8.2 Log Compaction Details

log compaction是由一个后台的线程池log cleaner来做的,不会block前台的produce/consume。同时也有一个用户参数来限制compaction的I/O带宽占用。一次log clean包含以下四步:

  1. 选择最大的比例:log head/log tail
  2. 用一个哈希表对log head中的每个key进行存储
  3. 从头到尾重新copy数据到一个新的位置,那些老keys会被直接删除,新的位置写满了1个segment file就会copy回去,所以只会有1个Segment file的额外空间占用。

9. Quota

这个是kafka在消费组/消费者之间的调度系统,放止某些消费者故意捣乱频繁拉取数据,占据了大量broker的资源而产生的。可以按照带宽/请求量进行分配,这里偏运维不细说,遇到再补充。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容