kafka zookeeper 部署 zookeeper在kafka集群中的作用_zookeeper


 kafaka集群中的 Broker 和 Consumer 都需要连接 Zookeeper,而 Producer 则直接连接 Broker。zookeeper在kafka中的作用如下:

 1、Broker的注册
  Broker是分布式部署的,并且相互之间是独立的。但是需要有一个注册系统能够将整个集群中的Broker管理起来,此时就使用到了Zookeeper。在Zookeeper上会有一个专门用来进行Broker服务器列表记录的节点:

/brokers/ids

  每个Broker在启动时,都会到Zookeeper上进行注册,即到 /brokers/ids 下创建属于自己的节点,如/brokers/ids/[0…N]。
  Kafka使用了全局唯一的数字来指代每个Broker服务器,不同的Broker必须使用不同的 Broker ID 进行注册。创建完节点后,每个 Broker 就会将自己的IP地址和端口信息记录到该节点中去。其中,Broker创建的节点类型是临时节点,一旦Broker宕机,则对应的临时节点也会被自动删除。

 2、Topic的注册
  在Kafka中,同一个Topic的消息会被分成多个分区并将其分布在多个Broker上,这些分区信息及与Broker的对应关系也都是由Zookeeper在维护,由专门的节点来记录,如:

/borkers/topics

  Kafka中每个Topic都会以/brokers/topics/[topic]的形式被记录,如/brokers/topics/login和/brokers/topics/search等。Broker服务器启动后,会到对应Topic节点(/brokers/topics)上注册自己的Broker ID并写入针对该Topic的分区总数,如/brokers/topics/login/3->2,这个节点表示Broker ID为3的一个Broker服务器,对于"login"这个Topic的消息,提供了2个分区进行消息存储。 同样,这个分区节点也是临时节点。

 3、生产者负载均衡
  由于同一个Topic的消息会被分区并将其分布在多个Broker上,因此,生产者需要将消息合理地发送到这些分布式的Broker上,那么如何实现生产者的负载均衡,Kafka支持传统的四层负载均衡,也支持Zookeeper方式实现的负载均衡。
  (1) 四层负载均衡:根据生产者的IP地址和端口来为其确定一个相关联的Broker。通常,一个生产者只会对应单个Broker,然后该生产者产生的消息都发往该Broker。这种方式逻辑简单,每个生产者不需要同其他系统建立额外的TCP连接,只需要和Broker维护单个TCP连接即可。但是,其无法做到真正的负载均衡,因为实际系统中的每个生产者产生的消息量及每个Broker的消息存储量都是不一样的,如果有些生产者产生的消息远多于其他生产者的话,那么会导致不同的Broker接收到的消息总数差异巨大,同时,生产者也无法实时感知到Broker的新增和删除。
  (2) 使用Zookeeper进行负载均衡:由于每个Broker启动时,都需完成Broker的注册过程,生产者会通过该节点的变化来动态地感知到Broker服务器列表的变更,这样就可以实现动态的负载均衡机制。

 4、消费者负载均衡
  与生产者类似,Kafka中的消费者同样需要进行负载均衡来实现多个消费者合理地从对应的Broker服务器上接收消息,每个消费者分组包含若干消费者,每条消息都只会发送给分组中的一个消费者,不同的消费者分组消费自己特定的Topic下面的消息,互不干扰。

 5、分区与消费者的关系
  消费组 (Consumer Group):
  consumer group 下有多个 Consumer(消费者)。对于每个消费者组 (Consumer Group),Kafka都会为其分配一个全局唯一的Group ID,Group 内部的所有消费者共享该 ID。订阅的topic下的每个分区只能分配给某个 group 下的其中一个consumer(当然该分区还可以被分配给其他group)。同时,Kafka为每个消费者分配一个Consumer ID,通常采用"Hostname:UUID"形式表示。
  在Kafka中,规定了每个消息分区只能被同组的一个消费者进行消费,因此,需要在 Zookeeper 上记录消息分区与 Consumer 之间的关系,每个消费者一旦确定了对一个消息分区的消费权力,需要将其Consumer ID 写入到 Zookeeper 对应消息分区的临时节点上,例如:

/consumers/[group_id]/owners/[topic]/[broker_id-partition_id]

  其中,[broker_id-partition_id]就是一个 消息分区的标识,节点内容就是该消息分区上消费者的Consumer ID。

 6、消费者注册
  消费者服务器在初始化启动时加入消费者分组的步骤如下:
   ①注册到消费者分组。每个消费者服务器启动时,都会到Zookeeper的指定节点下创建一个属于自己的消费者节点,例如/consumers/[group_id]/ids/[consumer_id],完成节点创建后,消费者就会将自己订阅的Topic信息写入该临时节点。
   ②对消费者分组中的消费者的变化注册监听。每个消费者都需要关注所属消费者分组中其他消费者服务器的变化情况,即对/consumers/[group_id]/ids节点注册子节点变化的Watcher监听,一旦发现消费者新增或减少,就触发消费者的负载均衡(即分区重新分配)。
   ③对Broker服务器变化注册监听。消费者需要对/broker/ids/[0-N]中的节点进行监听,如果发现Broker服务器列表发生变化,那么就根据具体情况来决定是否需要进行消费者负载均衡。
   ④进行消费者负载均衡。为了让同一个Topic下不同分区的消息尽量均衡地被多个消费者消费而进行消费者与消息分区分配的过程,通常,对于一个消费者分组,如果组内的消费者服务器发生变更或Broker服务器发生变更,会触发消费者负载均衡。

 7、Leader的选举

  Kafka 集群中有一个 broker 会被选举为 Controller,负责管理集群中 broker 的上下线、所有 topic 的分区副本分配和 leader 选举等工作。Controller 的管理工作是依赖于 zookeeper 的,下图为 partition的leader选举示意图:其他的选举与其类似

kafka zookeeper 部署 zookeeper在kafka集群中的作用_负载均衡_02


 总起来说,Zookeeper 在 Kafka 集群中的作用有以下:

  ①组件的注册:生产者的注册、消费者的注册、topic 的注册

  ②负载均衡:生产者的负载均衡、消费者的负载均衡

  ③分区的分配:各个 topic 的各个分区分配给哪个消费者消费

  ④Leader的选举:partition 中 leader 的选举等