您的位置:首页 > 新闻 > 会展 > Kafka简介

Kafka简介

2025/1/15 13:14:11 来源:https://blog.csdn.net/weixin_62587914/article/details/141570017  浏览:    关键词:Kafka简介

Kafka简介

1.1消息队列
  • 消息队列—用于存放消息的队列
  • 消息队列大多时候用于临时性存储
  • 可以用于系统解耦
  • 流量削峰
  • 消息队列有发布/订阅模型和点对点模型
1.1.2发布/订阅者模型的特点
  • 每个消息可以有多个订阅者
  • 发布者和订阅者之间有时间上的依赖,针对某个Topic(主题),必须有一个对应的订阅者才可以消费此Topic的主题
  • 为了可以消费消息,订阅者需要提前订阅该角色的Topic,并保持在线运行

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

二、Kafka介绍

2.1概念

Kafka是最初由Linkedin公司开发,是一个分布式、支持分区的(partition)、多副本的 (replica),基于zookeeper协调的分布式消息系统,它的最大的特性就是可以实时的处理 大量数据以满足各种需求场景:比如基于hadoop的批处理系统、低延迟的实时系统、 Storm/Spark流式处理引擎,web/nginx日志、访问日志,消息服务等等,用scala语言编 写,Linkedin于 2010 年贡献给了Apache基金会并成为顶级开源 项目

2.2使用场景
  • 日志收集:一个公司可以用Kafka收集各种服务的log,通过kafka以统一接口服务的方式 开放给各种consumer,例如hadoop、Hbase、Solr等。
  • 消息系统:解耦和生产者和消费者、缓存消息等。
  • 用户活动跟踪:Kafka经常被用来记录web用户或者app用户的各种活动,如浏览网⻚、 搜索、点击等活动,这些活动信息被各个服务器发布到kafka的topic中,然后订阅者通过 订阅这些topic来做实时的监控分析,或者装载到hadoop、数据仓库中做离线分析和挖掘。
  • 运营指标:Kafka也经常用来记录运营监控数据。包括收集各种分布式应用的数据,生产 各种操作的集中反馈,比如报警和报告
2.3Kafka相关术语
名称解释
Broker消息中间件处理节点,⼀个Kafka节点就是⼀个broker,⼀个或者多个Broker可以组成⼀个Kafka集群
TopicKafka根据topic对消息进⾏归类,发布到Kafka集群的每条消息都需要指定⼀个topic
Producer消息⽣产者,向Broker发送消息的客户端
Consumer消息消费者,从Broker读取消息的客户端
ConsumerGroup每个Consumer属于⼀个特定的Consumer Group,⼀条消息可以被多个不同的Consumer Group消费,但是⼀个Consumer Group中只能有⼀个Consumer能够消费该消息
Partition物理上的概念,⼀个topic可以分为多个partition,每个partition内部消息是有序的

因此,从一个较高层次看,producer通过网络发送信息到kafka集群,然后consumer来进行消费,如下图:

image-20240305141201321

服务端(Broker)和客户端(producer、consumer)之间听通过TCP协议来完成

三、Kafka基本使用

1、安装前环境准备
  • 安装jdk

  • 安装zk

  • 官网下载kafka的压缩包:http://kafka.apache.org/downloads

  • 解压缩至如下路径

    /usr/local/kafka/Copy to clipboardErrorCopied
    
  • 修改配置文件:/usr/local/kafka/kafka2.11-2.4/config/server.properties

    #broker.id属性在kafka集群中必须要是唯一
    broker.id= 0
    #kafka部署的机器ip和提供服务的端口号
    listeners=PLAINTEXT://192.168.65.60:9092
    #kafka的消息存储文件
    log.dir=/usr/local/data/kafka-logs
    #kafka连接zookeeper的地址
    zookeeper.connect= 192.168.65.60:2181
    
2、启动kafka服务器

进入到bin目录下。使用命令来启动

./kafka-server-start.sh -daemon../config/server.propertiesCopy to clipboardErrorCopied

验证是否启动成功:

进入到zk中的节点看id是 0 的broker有没有存在(上线)

ls /brokers/ids/Copy to clipboardErrorCopied

server.properties核心配置详解:

PropertyDefaultDescription
broker.id0每个broker都可以⽤⼀个唯⼀的⾮负整数id进⾏标识;这个id可以作为broker的“名字”,你可以选择任意你喜欢的数字作为id,只要id是唯⼀的即可。
log.dirs/tmp/kafka-logskafka存放数据的路径。这个路径并不是唯⼀的,可以是多个,路径之间只需要使⽤逗号分隔即可;每当创建新partition时,都会选择在包含最少partitions的路径下进⾏。
listenersPLAINTEXT://192.168.65.60:9092server接受客户端连接的端⼝,ip配置kafka本机ip即可
zookeeper.connectlocalhost:2181zooKeeper连接字符串的格式为:hostname:port,此处hostname和port分别是ZooKeeper集群中某个节点的host和port;zookeeper如果是集群,连接⽅式为hostname1:port1, hostname2:port2,hostname3:port3
log.retention.hours168每个⽇志⽂件删除之前保存的时间。默认数据保存时间对所有topic都⼀样。
num.partitions1创建topic的默认分区数
default.replication.factor1⾃动创建topic的默认副本数量,建议设置为⼤于等于2
min.insync.replicas1当producer设置acks为-1时,min.insync.replicas指定replicas的最⼩数⽬(必须确认每⼀个repica的写数据都是成功的),如果这个数⽬没有达到,producer发送消息会产⽣异常
delete.topic.enablefalse是否允许删除主题
3、创建主题Topic

topic是什么概念?topic可以实现消息的分类,不同消费者订阅不同的topic。

image-20240305141504089

执行以下命令创建名为“test”的topic,这个topic只有一个partition,并且备份因子也设置为1

./kafka-topics.sh --create --zookeeper localhost:2181 --replication-factor 1 --partitions 1 --topic testCopy to clipboardErrorCopied

查看当前kafka内有哪些topic

./kafka-topics.sh --list --zookeeper localhost:2181
4、发送信息

kafka自带了一个producer命令客户端,可以从本地文件中读取内容,或者我们也可以以命令行中直接输入内容,并将这些内容以消息的形式发送到kafka集群中。在默认情况下,每一个行会被当做成一个独立的消息。使用kafka的发送消息的客户端,指定发送到的kafka服务器地址和topic

./kafka-console-producer.sh --broker-list localhost:9092 --topic test
5、消费消息

对于consumer,kafka同样也携带了一个命令行客户端,会将获取到内容在命令中进行输 出, 默认是消费最新的消息 。使用kafka的消费者消息的客户端,从指定kafka服务器的指定 topic中消费消息

方式一:从最后一条消息的offset(偏移量+1)开始消费,偏移量其实就是这条消息在Topic中的位置

./kafka-console-consumer.sh --bootstrap-server localhost:9092 --topic testCopy to clipboardErrorCopied

方式二:从头开始消费

./kafka-console-consumer.sh --bootstrap-server localhost:9092 --from-beginning --topic testCopy to clipboardErrorCopied
  • 消息会被存储
  • 消息是顺序存储
  • 消息是有偏移量的
  • 消费时可以指明偏移量进行消费指定消息
  • 偏移量其实就是消息在Topic中位置

四、Kafka中的关键细节

1、消息的顺序存储

消息的发送方会把消息发送到broker中,broker会存储消息,消息是按照发送的顺序进行存储。因此消费者在消费消息时可以指明主题中消息的偏移量。默认情况下,是从最后一个消息的下一个偏移量开始消费。

2、单播消息的实现

单播消息:一个消费组里 只会有一个消费者能消费到某一个topic中的消息。于是可以创建多个消费者,这些消费者在同一个消费组中。

./kafka-console-consumer.sh --bootstrap-server 10.31.167.10:9092 --consumer-property group.id=testGroup --topic testCopy to clipboardErrorCopied
3、多播消息的实现

在一些业务场景中需要让一条消息被多个消费者消费,那么就可以使用多播模式

要想实现一条消息被多个小消费者消费,可以将这些消费者分成不同的消费组既可,也就是多播模式

kafka实现多播,只需要让不同的消费者处于不同的消费组即可。

./kafka-console-consumer.sh --bootstrap-server 10.31.167.10:9092 --consumer-property group.id=testGroup1 --topic test./kafka-console-consumer.sh --bootstrap-server 10.31.167.10:9092 --consumer-property group.id=testGroup2 --topic testCopy to clipboardErrorCopied
4、查看消费组及信息
# 查看当前主题下有哪些消费组
./kafka-consumer-groups.sh --bootstrap-server 10.31.167.10:9092 --list
# 查看消费组中的具体信息:比如当前偏移量、最后一条消息的偏移量、堆积的消息数量
./kafka-consumer-groups.sh --bootstrap-server localhost:9092 --describe --group testGroupCopy to clipboardErrorCopied

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • Currennt-offset: 当前消费组的已消费偏移量
  • Log-end-offset: 主题对应分区消息的结束偏移量(HW)
  • Lag: 当前消费组未消费的消息数

五、主题、分区的概念

1、主题Topic

主题Topic其实就是一个类别的名称

2、partition分区

partition其实就是Topic的一个子单位,一个主题中的消息量可能会非常巨大,通过分区的设置,用分布式来存储这些消息,比如一个Topic创建了三个partition,那么Topic中的消息就会被分别存储在这三个分区当中

  • 分区存储,可以解决统一存储文件过大的问题
  • 增大kafka的吞吐量,可以并行写

image-20240305142854117

  • 为一个主题创建多个分区

    ./kafka-topics.sh --create --zookeeper localhost:2181 --partitions 2 --topic test1
    
  • 查看Topic的分区信息

    ./kafka-topics.sh --describe --zookeeper localhost:2181 --topic test1
    
  • 分区的作用

    • 可以分布式存储消息
    • 可以并行写

小细节:

kafaka内部会自己创建一个Topic,consumer_offsets,每个消费者都会维护自己消费的主题和偏移量,消费者定期将自己消费分区的offset提交给kafka内部topic:__consumer_offsets,提交过去的 时候,key是consumerGroupId+topic+分区号,value就是当前offset的值,kafka会定 期清理topic里的消息,最后就保留最新的那条数据 因为__consumer_offsets可能会接收高并发的请求,kafka默认给其分配 50 个分区(可以 通过offsets.topic.num.partitions设置),这样可以通过加机器的方式抗大并发。 通过如下公式可以选出consumer消费的offset要提交到__consumer_offsets的哪个分区 公式:hash(consumerGroupId) % __consumer_offsets主题的分区数

  • 这样做的作用是,假设一个Topic里有100条消息,如果一个消费者消费了50条消息后挂掉,下一个消费者来消费这个Topic时,就会从consumer_offsets中获取上一个消费者的offset,然后继续消费消息

六、kafka集群及副本的概念

1、搭建kafka集群3个Broker

准备 3 个server.properties文件

每个文件中的这些内容要调整

  • server.properties

    broker.id= 0
    listeners=PLAINTEXT://192.168.65.60:
    log.dir=/usr/local/data/kafka-logsCopy to clipboardErrorCopied
    
  • server1.properties

    broker.id= 1
    listeners=PLAINTEXT://192.168.65.60:
    log.dir=/usr/local/data/kafka-logs-Copy to clipboardErrorCopied
    
  • server2.properties

    broker.id= 2
    listeners=PLAINTEXT://192.168.65.60:
    log.dir=/usr/local/data/kafka-logs-Copy to clipboardErrorCopied
    

    使用以下命令启动3台服务器

./kafka-server-start.sh -daemon../config/server0.properties
./kafka-server-start.sh -daemon../config/server1.properties
./kafka-server-start.sh -daemon../config/server2.propertiesCopy to clipboardErrorCopied

搭建完后通过查看zk中的/brokers/ids 看是否启动成功

2、副本的概念

副本就是对分区(partition)的备份,不同的副本会被部署在不同的broker上

通过查看主题信息,其中的关键数据:

  • replicas:当前副本存在的broker节点
  • leader:副本里的概念
    • 每个partition都有一个broker作为leader。
    • 消息发送方要把消息发给哪个broker?就看副本的leader是在哪个broker上面。副本里的leader专⻔用来接收消息。
    • 接收到消息,其他follower通过poll的方式来同步数据。
  • follower:leader处理所有针对这个partition的读写请求,而follower被动复制leader,不提供读写(主要是为了保证多副本数据与消费的一致性),如果leader所在的broker挂掉,那么就会进行新leader的选举

通过kill掉leader后再查看主题情况

# kill掉leader
ps -aux | grep server.properties
kill 17631
# 查看topic情况
./kafka-topics.sh --describe --zookeeper 172.16.253.35:2181 --topic my-replicated-topicCopy to clipboardErrorCopied

isr: 可以同步的broker节点和已同步的broker节点,存放在isr集合中。

副本的作用
  1. 提供数据冗余:副本的存在意味着同一分区下的数据有多个拷贝,分散保存在不同的Broker上。这种冗余备份能够对抗部分Broker宕机带来的数据不可用问题,即使系统部分组件失效,系统依然能够继续运转,从而增加了整体可用性以及数据持久性。
  2. 实现故障的自动转移:当Kafka集群中某个Broker失效时,副本机制能够实现故障的自动转移,保证服务依然可用。
  3. 提高读性能:通过增加机器(即增加副本)的方式,Kafka能够支持横向扩展,提升读性能,进而提高读操作吞吐量。
3、broker、主题、分区、副本
  • kafka集群中由多个broker组成
  • 一个broker中存放一个Topic的不同partition-----副本
4、kafka集群消息的发送
./kafka-console-producer.sh --broker-list localhost:9092,localhost:9093,localhost:9094 --topic my-replicated-topic
5、kafka集群消息的消费
./kafka-console-consumer.sh --bootstrap-server localhost:9092,localhost:9093,localhost:9094 --from-beginning --topic my-replicated-topic
6、关于分区消费组消费者的细节

image-20240305154804259

图中Kafka集群有两个broker,每个broker中有多个partition。一个partition只能被一个消费组里的某一个消费者消费,从而保证消费顺序。Kafka只在partition的范围内保证消息消费的局部顺序性,不能在同一个topic中的多个partition中保证总的消费顺序性。一个消费者可以消费多个partition。

  • 消费组中消费者的数量不能比一个topic中的partition数量多,否则多出来的消费者消费不到消息
  • 一个partition只能被一个消费组中的一个消费者消费,,目的是为了保证消费的顺序性
7、broker、主题、分区、副本之间的关系

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • 副本的作用其实就是备份,为主题中的分区创建多个备份,多个副本在kafka集群的多个broker中,会有一个副本作为leader,其他是follower
  • leader:是副本中的概念,生产者和消费者在读和写的时候,都会在leader所在的副本中进行,读写完成之后,leader会向其他两个副本所在的broker进行同步,如果当leader挂掉之后,就会从这个集Isr中重新选举一个leader
  • Replicas:接收leader同步的数据
  • Isr:是一个集合,用于保存可以同步的broker节点和已经同步的broker节点,作用就是,如果当leader挂掉之后,就会从这个集合中重新选举一个leader,如果isr中的节点性能较差,则会被踢出isr集合

一个集群中有多个broker,创建主题时可以指明主题有多个分区(把Topic中的消息拆分到不同的分区中),一个分区可以有多个副本,不同的副本存放在不同的broker中

七、Kafka的Java客户端----生产者

1.引入依赖
<dependency><groupId>org.apache.kafka</groupId><artifactId>kafka-clients</artifactId><version>2.4.1</version>
</dependency>
2.生产者发送消息的基本实现
//消息的发送方
public class MyProducer {private final static String TOPIC_NAME = "my-replicated-topic";public static void main(String[] args) throws ExecutionException,InterruptedException {
Properties props = new Properties();
props.put(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG,"10.31.167.10:9092,10.31.167.10:9093,10.31.167.10:9094");
//把发送的key从字符串序列化为字节数组
props.put(ProducerConfig.KEY_SERIALIZER_CLASS_CONFIG,StringSerializer.class.getName());
//把发送消息value从字符串序列化为字节数组
props.put(ProducerConfig.VALUE_SERIALIZER_CLASS_CONFIG,StringSerializer.class.getName());Producer<String, String> producer = new KafkaProducer<String,String>(props);Order order = new Order((long) i, i);
ProducerRecord<String, String> producerRecord = new ProducerRecord<String, String>(TOPIC_NAME, order.getOrderId().toString(), JSON.toJSONString(order));
RecordMetadata metadata = producer.send(producerRecord).get();
//=====阻塞=======
System.out.println("同步方式发送消息结果:" + "topic-" +metadata.topic() + "|partition-"+ metadata.partition() + "|offset-" +metadata.offset());
3.发送消息到指定分区上
ProducerRecord<String, String> producerRecord = new ProducerRecord<String, String>(TOPIC_NAME, 0 , order.getOrderId().toString(), JSON.toJSONString(order));
4.如果未指定分区,则会按照key来计算出往哪个分区上发送
//未指定发送分区,具体发送的分区计算公式:hash(key)%partitionNum
ProducerRecord<String, String> producerRecord = new ProducerRecord<String, String>(TOPIC_NAME, order.getOrderId().toString(), JSON.toJSONString(order));
5.同步发送

生产者同步发送消息,在收到kafka的ack之前会一直处于阻塞状态

//等待消息发送成功的同步阻塞方法
RecordMetadata metadata = producer.send(producerRecord).get();
System.out.println("同步方式发送消息结果:" + "topic-" +metadata.topic() + "|partition-"+ metadata.partition() + "|offset-" +metadata.offset());
6.异步发送

生产者异步发送消息,在发送完成之后,不需要等待broker给回复,直接执行下面的业务逻辑,同时异步调用kafka提供的callback,让broker用异步的方式,告知生产者消息发送的结果

//要发送 5 条消息
Order order = new Order((long) i, i);
//指定发送分区
ProducerRecord<String, String> producerRecord = new ProducerRecord<String, String>(TOPIC_NAME, 0 , order.getOrderId().toString(),JSON.toJSONString(order));
//异步回调方式发送消息
producer.send(producerRecord, new Callback() {
public void onCompletion(RecordMetadata metadata, Exception exception) {
if (exception != null) {System.err.println("发送消息失败:" +exception.getStackTrace());
}
if (metadata != null) {
System.out.println("异步方式发送消息结果:" + "topic-" +metadata.topic() + "|partition-"+ metadata.partition() + "|offset-" + metadata.offset());}}
});
  • 在实际应用过程中,更多的会使用同步发送方式去发送消息,使用异步发送方式虽然可以小幅度的提升效率,但是可能会导致消息的丢失
7.关于生产者ack参数配置

在同步发送方式下,可以手动配置

  • ack=0,表示生产者不需要等待任何broker确认收到消息的回复,就可以继续发送下一条,性能最高,但容易丢失消息
  • ack=1,至少要等待leader已经成功将数据写入本地log中,但是不需要等待所有follower是否成功写入,就可以继续发送下一条消息。这种情况下,如果follower没有备份消息,而此时leader又挂掉,则消息就会丢失
  • ack=-1/all:需要等待min.insync.replicas(默认为1,推荐配置大于等于2)这个参数配置的副本个数都成功写入日志,这种策略会保证只要有一个备份存活,数据就不会丢失,这是最强的数据保证

image-20240307190552304

kafka的关键能力
  • 发布和订阅数据流
  • 以容错的持久化方式存储数据流
  • 处理数据流
kafka主要用于:
  • 建立实时数据流管道,也就是可以通过kafka进行消息接收与转发

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com