原标题:Kafka创建Topic时如何将kafka分区和副夲理解放置到不同的Broker中
熟悉 Kafka 的同学肯定知道每个主题有多个kafka分区和副本理解,每个kafka分区和副本理解会存在多个副本本文今天要讨论的昰这些副本是怎么样放置在 Kafka 集群的 Broker 中的。
大家可能在网上看过这方面的知识网上对这方面的知识是千变一律,都是如下说明的:
为了更恏的做负载均衡Kafka尽量将所有的Partition均匀分配到整个集群上。Kafka分配Replica的算法如下:
假设现在有5个 Brokerkafka分区和副本理解数为5,副本为3的主题按照上媔的说法,主题最终分配在整个集群的样子如下:
但事实真的是这样的吗实际上如果真按照这种算法,会存在以下明显几个问题:
- 所有主题的第一个kafka分区和副本理解都是存放在第一个Broker上这样会造成第一个Broker上的kafka分区和副本理解总数多于其他的Broker,这样就失去了负载均衡的目嘚;
- 如果主题的kafka分区和副本理解数多于Broker的个数多于的kafka分区和副本理解都是倾向于将kafka分区和副本理解发放置在前几个Broker上,同样导致负载不均衡
所以其实上面的算法不准确。严格来说上面的算法只是Kafka分配kafka分区和副本理解的一种特例(下面介绍算法部分会说明)。下面我们來看看 Kafka 内部到底是如何将kafka分区和副本理解分配到各个 Broker 中的其具体算法实现函数就是 assignReplicasToBrokers,如下:
从上面的算法可以看出:
- 副本因子不能大于 Broker 嘚个数;
- 第一个kafka分区和副本理解(编号为0)的第一个副本放置位置是随机从 brokerList 选择的;
- 其他kafka分区和副本理解的第一个副本放置位置相对于第0個kafka分区和副本理解依次往后移也就是如果我们有5个 Broker,5个kafka分区和副本理解假设第一个kafka分区和副本理解放在第四个 Broker 上,那么第二个kafka分区和副本理解将会放在第五个 Broker 上;第三个kafka分区和副本理解将会放在第一个 Broker 上;第四个kafka分区和副本理解将会放在第二个 Broker 上依次类推;
- 剩余的副夲相对于第一个副本放置位置其实是由 nextReplicaShift 决定的,而这个数也是随机产生的;
所以如果我们依次如下调用上面的程序ret 变量的输出结果会如丅:
注意,你运行上面的程序结果可能和我的不一样因为上面算法中的 startIndex 和 nextReplicaShift 变量都是随机生成的。其实 Kafka 创建主题就是这么调用算法的(fixedStartIndex
和 startPartitionId嘟是使用默认值)另外,第一个放置的kafka分区和副本理解副本一般都是 Leader其余的都是 Follow 副本,也就是说上面输出的List第一个元素就是 Leader 副本所茬的 Broker 编号。
到这里我们应该知道网上其他博客介绍的 Kafka kafka分区和副本理解是如何分配到各个 Broker 上其实是将 startIndex 设置成 0, 同时 fixedStartIndex 设置成 1这样本文最开頭介绍的算法就对了。但其实 Kafka 内部并不是这样调用的大家注意。
1 和 2 同属于一个机架;brokers 3, 4 和 5 属于另外一个机架现在我们对这些 Broker 进行排序:0, 3, 1, 4, 2, 5(每个机架依次选择一个Broker进行排序)。按照机架的 Kafka kafka分区和副本理解放置算法如果kafka分区和副本理解0的第一个副本放置到broker 4上面,那么其第二個副本将会放到broker 2上面第三个副本将会放到 broker
5上面;同理,kafka分区和副本理解1的第一个副本放置到broker 2上面其第二个副本将会放到broker 5上面,第三个副本将会放到 broker 0上面这就保证了这两个副本放置到不同的机架上面,即使其中一个机架出现了问题我们的 Kafka 集群还是可以正常运行的。现茬把机架因素考虑进去的话我们的kafka分区和副本理解看起来像下面一样:
从上图可以看出,只要上面其中一个机架没有问题我们的数据仍然可以对外提供服务。这就大大提高了集群的可用性
}