大家好,我是 yes。
这是我的第三篇Kafka源码分析文章,前两篇讲了日志段的读写和二分算法在kakfa索引上的应用
今天来讲讲 Kafka Broker
端处理请求的全流程,剖析下底层的网络通信是如何实现的、 Reactor
在 kafka
上的应用。
再说说社区为何在2.3版本将请求类型划分成两大类,又是如何实现两类请求处理的优先级。
叨叨
不过在进入今天主题之前我想先叨叨几句,就源码这个事儿,不同人有不同的看法。
有些人听到源码这两个词就被吓到了,这么多代码怎么看。奔进去就像无头苍蝇,一路断点跟下来,跳来跳去,算了拜拜了您嘞。
而有些人觉得源码有啥用,看了和没看一样,看了也用不上。
其实上面两种想法我都有过,哈哈哈。那为什么我会开始看 Kafka
源码呢?
其实就是我有个同事在自学 go
,然后想用 go 写个消息队列,在画架构图的时候就来问我,这消息队列好像有点东西啊,消息收发,元数据管理,消息如何持久一堆问题过来,我直呼顶不住。
这市面上 Kafka
、 RocketMQ
都是现成的方案,于是乎我就看起了源码。
所以促使我看源码的初始动力,竟然是为了在同事前面装逼!!
我是先看了 RocketMQ
,因为毕竟是 Java
写的,而 Kafka Broker
都是 scala
写的。
梳理了一波 RocketMQ
之后,我又想看看 Kafka
是怎么做的,于是乎我又看起了 Kafka
。
在源码分析之前我先总结性的说了说 Kafka
底层的通信模型。应对面试官询问 Kafka
请求全过程已经够了。
其实源码分析在手机上看效果欠佳,建议电脑端打开观看。
Reactor模式
在扯到 Kafka
之前我们先来说说 Reactor模式
,基本上只要是底层的高性能网络通信就离不开 Reactor模式
。像Netty、 Redis 都是使用 Reactor模式
。
像我们以前刚学网络编程的时候以下代码可是非常的熟悉,新来一个请求,要么在当前线程直接处理了,要么新起一个线程处理。
在早期这样的编程是没问题的,但是随着互联网的快速发展,单线程处理不过来,也不能充分的利用计算机资源。
而每个请求都新起一个线程去处理,资源的要求就太高了,并且创建线程也是一个重操作。
说到这有人想到了,那搞个线程池不就完事了嘛,还要啥 Reactor
。
池化技术确实能缓解资源的问题,但是池子是有限的,池子里的一个线程不还是得候着某个连接,等待指示嘛。现在的互联网时代早已突破 C10K
了。
因此引入的 IO多路复用
,由 一个线程来监视一堆连接 ,同步等待一个或多个IO事件的到来,然后将事件分发给对应的 Handler
处理,这就叫 Reactor模式
。
网络通信模型的发展如下
单线程 => 多线程 => 线程池 => Reactor模型
Kafka所采用的 Reactor模型
如下
Kafka Broker 网络通信模型
简单来说就是,Broker 中有个 Acceptor(mainReactor)
监听新连接的到来,与新连接建连之后轮询选择一个 Processor(subReactor)
管理这个连接。
而 Processor
会监听其管理的连接,当事件到达之后,读取封装成 Request
,并将 Request
放入共享请求队列中。
然后IO线程池不断的从该队列中取出请求,执行真正的处理。处理完之后将响应发送到对应的 Processor
的响应队列中,然后由 Processor
将 Response
返还给客户端。
每个 listener
只有一个 Acceptor线程
,因为它只是作为新连接建连再分发,没有过多的逻辑,很轻量,一个足矣。
Processor
在Kafka中称之为 网络线程 ,默认网络线程池有3个线程,对应的参数是 num.network.threads
。并且可以根据实际的业务动态增减。
还有个 IO 线程池,即 KafkaRequestHandlerPool
,执行真正的处理,对应的参数是 num.io.threads
,默认值是 8。IO线程处理完之后会将 Response
放入对应的 Processor
中,由 Processor
将响应返还给客户端。
可以看到网络线程和IO线程之间利用的经典的生产者 - 消费者模式,不论是用于处理Request的共享请求队列,还是IO处理完返回的Response。
这样的好处是什么? 生产者和消费者之间解耦了,可以对生产者或者消费者做独立的变更和扩展 。并且可以平衡两者的处理能力,例如消费不过来了,我多加些IO线程。
如果你看过其他中间件源码,你会发现生产者-消费者模式真的是太常见了,所以面试题经常会有手写一波生产者-消费者。
源码级别剖析网络通信模型
Kafka 网络通信组件主要由两大部分构成:
SocketServer和 KafkaRequestHandlerPool 。
SocketServer
可以看出 SocketServer
旗下管理着, Acceptor 线程
、 Processor 线程
和 RequestChannel
等对象。
data-plane
和 control-plane
稍后再做分析,先看看 RequestChannel
是什么。
RequestChannel
关键的属性和方法都已经在下面代码中注释了,可以看出这个对象主要就是 管理 Processor
和 作为传输 Request
和 Response
的中转站 。
Acceptor
接下来我们再看看 Acceptor
可以看到它继承了 AbstractServerThread
,接下来再看看它run些啥
再来看看 accept(key)
做了啥
很简单,标准 selector
的处理,获取准备就绪事件,调用 serverSocketChannel.accept()
得到 socketChannel
,将 socketChannel
交给通过轮询选择出来的 Processor
,之后由它来处理IO事件。
Processor
接下来我们再看看 Processor
,相对而言比 Acceptor
复杂一些。
先来看看三个关键的成员
再来看看主要的处理逻辑。
可以看到 Processor
主要是将底层读事件IO数据封装成 Request
存入队列中,然后将IO线程塞入的 Response
,返还给客户端,并处理 Response
的回调逻辑。
KafkaRequestHandlerPool
IO线程池,实际处理请求的线程。
再来看看IO线程都干了些啥
很简单,核心就是不断的从 requestChannel
拿请求,然后调用handle处理请求。
handle
方法是位于 KafkaApis
类中,可以理解为通过 switch
,根据请求头里面不同的 apikey
调用不同的 handle
来处理请求。
我们再举例看下较为简单的处理 LIST_OFFSETS
的过程,即 handleListOffsetRequest
,来完成一个请求的闭环。
我用红色箭头标示了调用链。表明处理完请求之后是塞给对应的 Processor
的。
最后再来个更详细的总览图,把源码分析到的类基本上都对应的加上去了。
请求处理优先级
上面提到的 data-plane
和 control-plane
是时候揭开面纱了。这两个对应的就是数据类请求和控制类请求。
为什么需要分两类请求呢?直接在请求里面用key标明请求是要读写数据啊还是更新元数据不就行了吗?
简单点的说比如我们想删除某个topic,我们肯定是想这个topic马上被删除的,而此时producer还一直往这个topic写数据。
那这个情况可能是我们的删除请求排在第N个...等前面的写入请求处理好了才轮到删除的请求。实际上前面那些往这个topic写入的请求都是没用的, 平白的消耗资源 。
再或者说进行 Preferred Leader
选举时候, producer
将 ack
设置为 all
时候, 老leader
还在等着 follower
写完数据向他报告呢,谁知 follower
已经成为了 新leader
。
而通知它leader已经变更的请求由于被一堆数据类型请求堵着呢, 老leader
就傻傻的在等着,直到超时。
就是为了解决这种情况,社区将请求分为两类。
那如何让控制类的请求优先被处理?优先队列?
社区采取的是两套 Listener
,即数据类型一个 listener
,控制类一个 listener
。
对应的就是我们上面讲的 网络通信模型,在kafka中有两套! kafka通过两套监听变相的实现了请求优先级,毕竟数据类型请求肯定很多,控制类肯定少,这样看来控制类肯定比大部分数据类型先被处理!
迂回战术啊。
控制类的和数据类区别就在于:就一个 Porcessor线程
,并且请求队列写死的长度为20,社区觉得这样够了。
最后
看源码主要就是得耐心,耐心跟下去。然后再跳出来看。你会发现不过如此,哈哈哈。
前两篇由于授权给他人了,因此公众号上发不了,贴下连接,有兴趣的同学可以看下。
Kafka日志段读写分析:https://juejin.im/post/5ef6b94ae51d4534a1236cb0
Kafka索引在设计有什么亮点:https://juejin.im/post/5efdeae7f265da22d017e58d
我是yes,一个在互联网摸爬滚打且莫得感情的 工具 人。
以上所述就是小编给大家介绍的《面试官:说说Kafka处理请求的全流程》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!
猜你喜欢:- 面试官:说说 Kafka 处理请求的全流程
- Spring MVC源码(二) ----- DispatcherServlet 请求处理流程 面试必问
- 面试官:你连HTTP请求Post和Get都不了解?
- 疯了吧!面试官问一个TCP连接可以发多少个HTTP请求?
- 疯了疯了!面试官问一个 TCP 连接可以发多少个 HTTP 请求?
- 详解nginx的请求限制(连接限制和请求限制)
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
UNIX网络编程 卷2
W.Richard Stevens / 人民邮电出版社 / 2009-11 / 89.00元
《UNIX网络编程 卷2:进程间通信(英文版·第2版)》是一部UNIX网络编程的经典之作。进程间通信(IPC)几乎是所有Unix程序性能的关键,理解IPC也是理解如何开发不同主机间网络应用程序的必要条件。《UNIX网络编程 卷2:进程间通信(英文版·第2版)》从对Posix IPC和System V IPC的内部结构开始讨论,全面深入地介绍了4种IPC形式:消息传递(管道、FIFO、消息队列)、同......一起来看看 《UNIX网络编程 卷2》 这本书的介绍吧!