Kafka, 大数据

kafka-06 kafka生产者

分区策略

分区的原因

  • 方便在集群中扩展,每个partition可以通过调整以适应它所在的机器,而一个topic又可以有多个partition组成,因此整个集群就可以适应任意大小的数据了
  • 可以提高并发,因为可以以partition为单位读写了

分区的策略

将producer发送的数据封装成一个ProducerRecord对象

  • 指定partition的情况下,直接指明的值作为partition值
  • 没有指明partition值但有key的情况下,将key的hash值与topic的partition数进行取余得到partition值
  • 既没有partition值又没有key值的情况下,第一次调用时随机生成一个整数(后面每次调用在这个整数上自增),将这个值与topic可用的partition总数取余得到partition值,也就是常说的round-robin算法

数据可靠性保证

topic的每个partition收到producer发送的数据后,都会向producer发送ack(确认收到),producer收到ack就会继续进行下一轮的发送,否则重新发送数据

  • 第一种方案会超出大量数据冗余
  • 第二种方案的网络延迟会比较高,但网络延迟对kafka的影响比较小

ISR: in-sync replica set

  • 意为和leader保持同步的follower集合 不是真正意义上的follower
  • 当ISR中的follower完成数据的同步后,leader就会给producer发送ack
  • 如果follower长时间未向leader同步数据,则该follower将被剔除ISR,该时间阈值由replica.lag.time.max.ms参数设定
  • leader发生故障后就会从ISR中选举新的leader

ack应答机制

对于某些不太重要的数据,对数据的可靠性要求不是很高,没必要等ISR中的follower全部接收成功

三种可靠性级别可选:

acks参数配置:

acks:

  • 0 At Most Once 生产者每条消息发送一次(只管发消息)
    • producer不等待broker的ack,这一操作提供了一个最低的延迟,broker一接收到还没有写入磁盘就已经返回,当broker故障时有可能丢失数据
  • 1
    • producer等待broker的ack,partition的leader落盘成功后返回ack,如果follower同步成功之前leader故障,那么将会丢失数据
    • (leader写完还没同步数据挂掉了)
  • -1 At Least Once 每条消息最少发送一次 不会丢数据但是可能重复(all)
    • producer等待broker的ack,partition的leader和follower全部落盘成功后才返回ack,但是如果在follower同步完成后,broker发送ack之前,leader发生故障,那么会造成数据重复

故障处理细节

log文件中的hw(High Watermark)和leo(Log End Offset)

LEO:每个副本最大的offset

HW:消费者能见到的最大的offset,ISR队列中最小的LEO

  • follower故障
    • follower发生故障后被临时踢出ISR,待该follower恢复后,follwer会读取本地磁盘记录的上次HW,并将log文件高于hw的部分截取掉,从HW开始向leader进行同步
    • 等该follower的LEO大于等于该partition的hw,即follower追上leader之后,就可以重新加入ISR了
  • leader故障
    • leader发生故障之后会从ISR中选出一个新leader,之后为保证多个副本之间数据的一致性,其余的follower会将各自的log文件高于hw的部分截掉,然后从新的leader同步数据 (多退少补)
    • 这只能保证副本之间的数据一致性,不能保证数据不丢失或者不重复

hw保证消费一致性,存储一致性

Exactly Once

producer不论向server发送多少次重复数据,server端只会持久化一条

At Least Once + 幂等性 = Exactly Once

启动幂等性:Producer参数中enable.idompotence设置为true

在kafka中做去重

  • 开启幂等性的producer在初始化的时候会被分配一个PID (producer id)
  • 发往同一个partition的消息会附带Sequence Number
  • Broker会对做缓存,当具有相同主键的消息提交时,Broker只会持久化一条
  • 但是PID重启就会变化,同时不同的Partition也具有不同主键,所以幂等性无法保证跨分区跨会话的Exactly Once