5、Kafka面试相关问题

1. Kafka为什么这么快?

顺序I/O、索引、批量读写和文件压缩、零拷贝

1.1 顺序I/O

磁盘的构造如图。磁盘的盘片不停地旋转,磁头会在磁盘表面画出一个圆形轨迹,叫磁道。从内到外半径不同有很多磁道。然后又用半径线,把磁道分割成了扇区(两根射线之内的扇区组成扇面)。如果要读写数据, 必须找到数据对应的扇区,这个过程就叫寻址。

  • **随机I/O:**读写的多条数据在磁盘上是分散的,寻址会很耗时。
  • **顺序I/O:**读写的数据在磁盘上是集中的,不需要重复寻址的过程。

Kafka的message是不断追加到本地磁盘文件末尾的,而不是随机的写入 ,这使得 Kafka写入吞吐量得到了显著提升。

顺序IO到底有多快呢?下图显示,在一定条件下测试,磁盘的顺序读写可以达到53.2M每秒,比内存的随机读写还要快。

1.2 索引

在写入日志的时候,会建立关于Offset和时间的稀疏索引,提升了查找效率。

1.3 批量读写

Kafka无论是生产者发送消息还是消费者消费消息,都是批量操作的,大大提高读写性能。

1.4 零拷贝

操作系统虚拟内存的内核空间和用户空间:操作系统的虚拟内存分成了两块,一部分是内核空间,一部分是用户空间。这样就可以避免用户进程直接操作内核,保证内核安全

  • 内核空间:进程在内核空间可以执行任意命令,调用系统的一切资源
  • 用户空间:必须要通过---些系统接口才能向内核发出指令

如果用户要从磁盘读取数据(比如kafka消费消息),必须先把数据从磁盘拷贝到内核缓冲区 ,然后在从内核缓冲区到用户缓冲区最后才能返回给用户

DMA拷贝 :没有DMA技术的时候,拷贝数据的事情需要CPU亲自去做, 这个时候它没法干其他的事情,如果传输的数据量大那就有问题了。DMA技术叫做直接内存访问(Direct Memory Access), 其实可以理解为CPU给自己找了一个小弟帮它做数据搬运的事情。在进行I/O设备和内存的数据传输的时候,数据搬运的工作全部交给DMA控制器,解放了 CPU的双手。

理解了这两个东西之后,我们来看下传统的I/O模型:

kafka要消费消息,要先把数据从磁盘拷贝到内核缓冲区,然后拷贝到用户缓冲区,再拷贝到socket缓冲区,再拷贝到网卡设备。这里面发生了 4次用户态和内核态的切换和4次数据拷贝,2次系统函数的调用(read、write),这个过程是非常耗费时间的。怎么优化呢?

在Linux操作系统里面提供了一个sendfile函数(并不是所有操作系统都支持sendfile),可以实现"零拷贝"。这个时候就不需要经过用户缓冲区了,直接把数据拷贝到网卡(这里画的是支持SG-DMA拷贝的情况)。因为这个只有DMA拷贝,没有CPU拷贝,所以叫做"零拷贝"。零拷贝至少可以提高一倍的性能。

2. Kafka如何保证消息不丢失?

(1)Producer设置ACK级别,设置为-1,保证最高级别的可靠性,但性能会下降。

(2)Producer设置retries参数(重试次数)。

(3)Broker集群,有N个Broker,至少设置N个副本

java 复制代码
//表示分区副本的个数,replication.factor >1 当leader 副本挂了,
//follower副本会被选举为leader继续提供服务
replication.factor 

//表示 ISR 最少的副本数量,通常设置 min.insync.replicas >1,
//这样才有可用的follower副本执行替换,保证消息不丢失
min.insync.replicas 

//是否可以把非 ISR 集合中的副本选举为 leader 副本
unclean.leader.election.enable = false 

(4)Consumer把自动提交改成手动提交

java 复制代码
enable.auto.commit = false

(5)Producer发送消息时使用 producer.send(msg, callback) 方法,该方法支持回调函数,可以在消息发送失败后进行重试

3. Kafka如何避免重复消费?

Producer开启事务。

事务协调者(Transaction Coordinator)是Kafka Broker内置的一个组件。它负责管理生产者和消费者之间的事务性操作,确保消息生产和消费的原子性和一致性。

事务协调者的主要职责包括:

  1. 事务状态管理:跟踪每个事务的状态(例如,开启、准备提交、已提交、已中止等),并确保这些状态在Broker之间保持一致。
  2. 幂等性和恰好一次语义:通过与生产者协作,确保每条消息只能被写入一次,即使在网络故障或重试的情况下也不会重复写入。
  3. 跨分区事务支持:允许生产者在一个事务中向多个主题和分区发送消息,并保证这些操作要么全部成功,要么全部失败。
  4. 消费偏移量管理:在事务性消费场景下,协调者还负责管理消费者的偏移量提交,确保只有当所有相关操作都成功完成时才会更新偏移量。

Producer发送消息前,启用事务配置,设置参数 enable.idempotence = true 开启幂等性,设置唯一事务ID(可为UUID)。

Producer初始化事务,开启事务,执行业务逻辑并发送消息,提交事务。

在开启事务到提交事务操作之间进行错误捕获,发生异常后事务终止。

相关推荐
qq_124987075319 小时前
基于SpringCloud的分布式演唱会抢票系统(源码+论文+部署+安装)
分布式·spring·spring cloud·毕业设计·计算机毕业设计
vx-bot55566619 小时前
1024proxy现代对抗性环境下的分布式流量调度系统架构设计
分布式·系统架构
han_1 天前
从一道前端面试题,谈 JS 对象存储特点和运算符执行顺序
前端·javascript·面试
yumgpkpm1 天前
AI评判:信创替代对Cloudera CDH CDP Hadoop大数据平台有何影响?
大数据·hive·oracle·flink·kafka·hbase·cloudera
蝎子莱莱爱打怪1 天前
我的2025年年终总结
java·后端·面试
我的写法有点潮1 天前
JS中对象是怎么运算的呢
前端·javascript·面试
镜花水月linyi1 天前
Cookie、Session、JWT 的区别?
后端·面试
青莲8431 天前
Java内存模型(JMM)与JVM内存区域完整详解
android·前端·面试
青莲8431 天前
Java内存回收机制(GC)完整详解
java·前端·面试