一、Producer 的网络模型
我们前面几篇有说 Producer 发送流程的源码分析,但那个是大的轮廓,涉及到发送很多相关的内容,比如:
-
获取 topic 的 metadata 信息
-
key 和 value 的序列化
-
获取该 record 要发送到的 partition
-
向 RecordAccmulator 中追加 record 数据
-
唤醒 sender 线程发送 RecordBatch
那这篇老周主要来说下 Producer 的网络模型,这里直接给出 Producer 的网络模型图,如下:
从图中可以看出,KafkaProducer 相当于客户端,与 Sender 调用层交互,Sender 调用 NetworkClient,NetworkClient 调用 Selector,而 Selector 底层封装了 Java NIO 的相关接口。心中有了 Producer 的网络模型大致轮廓后,我们接下来就可以来分析 Producer 的网络模型。
二、Producer 与 Broker 的交互流程
我们在业务代码通过生产者 producer 调用 send 方法来发送消息,不难发现都是通过走 Producer 的实现类 KafkaProducer 的 send 方法:
2.1 org.apache.kafka.clients.producer.KafkaProducer#doSend
上面的两个 send 方法最终会走到 doSend 方法里来:
这块的源码老周在前两篇的 Producer 源码解析那一篇分析了的哈,这里主要说下与 Broker 通信的交互分析。主要有两点:
-
waitOnMetadata():请求 tp(topic-partition)元数据 metadata 更新,中间会调用 sender.wakeup()。
-
accumulator.append():将 record 对应的 tp 写入到 deque 中,如果该 tp 对应的 deque batch 是满了或者新建了一个 batch,则会调用 sender.wakeup()。
主要看下 sender.wakeup() 方法,主要作用就是将 Sender 线程从阻塞中唤醒。
2.2 org.apache.kafka.clients.producer.internals.Sender#wakeup
/**
* Wake up th