Kafka-消费者订阅主题和取消订阅

消费者订阅

消费者通过 subscribe()assign() 两种方式订阅主题

subscribe()

使用 subscribe() 可以订阅一个或多个主题,对于这个方法而言,可以以集合的方式订阅多个主题,也可以以正则表达式的形式订阅特定模式的主题。

subscribe 的几个重载方法如下:

public void subscribe(Collection<String> topics)
public void subscribe(Pattern pattern, ConsumerRebalanceListener listener)
public void subscribe(Pattern pattern)
public void subscribe(Pattern pattern, ConsumerRebalanceListener listener)

对于消费者以集合的方式订阅主题,如果前后两次订阅了不同的主题,以最后一次的订阅为准,前面的订阅都会失效,如:

consumer.subscribe(Arrays.asList("topic1"));
consumer.subscribe(Arrays.asList("topic2"));

上述的示例,最终订阅的是 topic2,由此可以看出来 topic1 订阅失效了。

如果以正则表达式的方式订阅主题,在之后的过程中,如果新创建了新的主题,并且主题的名称与正则表达式相匹配,那么这个消费者就可以消费到这个新添加的主题中的消息。

示例如下:

consumer.subscribe(Pattern.compile("superz-*"));

assign()

使用 assign() 可以指定订阅的主题分区

assign() 的方法定义如下:

public void assign(Collection<TopicPartition> partitions)

两者的区别

分区分配的区别:

  • subscribe() 是有 Kafka 内部算法为消费者自动分配分区
  • assign() 则需要开发者手动为消费者指定消费的分区

通过分区分配的区别可以看出来,采用 subscribe() 方式的订阅,多个消费者之间消费的消息不会重复,且所有消费者消费的消息是一个主题的全部消息;但使用 assign() 方式的订阅,在位移未提交的情况下,多个消费者订阅相同的主题分区,消费到的消息是完全一样的。

建议:assign()subscribe() 不要混用

因为 assign()subscribe() 订阅并配置 enable.auto.commit=true 的情况下,poll() 会提交偏移量,这样会造成 assign()subscribe() 都会对同一个主题分区提交偏移量,这样的偏移量对其中的一些订阅是有问题的。

集合订阅的方式subscribe和指定分区的订阅方式assign分表代表了两种不同的订阅状态。然而这两种状态是互斥的, 在一个消费者中只能使用其中的一种,否则会报出IllegalStateException 异常

java.lang.IllegalStateException : Subscription to topics , partitions and patter are mutually exclusive.

1) 通过subscribe()方法订阅主题具有消费者自动再均衡的功能,在多个消费者的情况下可以根据分区分配策略来自动分配各个消费者与分区的关系。当消费组内的消费者增加或减少时,分区分配关系会自动调整,以实现消费负载均衡及故障自动转移。

2) 而通过assign()方法订阅分区时, 是不具备消费者自动均衡的功能的,其实这一点从assign()方法的参数中就可以看出端倪,两种类型的subscribe()都有ConsumerRebalanceListener 类型参数的方法,而assign()方法却没有。

 

消费者取消订阅

在 KafkaConsumer 中可以使用 unsubscribe() 方法来取消主题的订阅。

这个方法可以取消如下的订阅方式:

  • subscribe(Collection) 方式实现的订阅
  • subscribe(Pattern) 方式实现的订阅
  • assign() 方式实现的订阅

使用方式的示例如下:

consumer.unsubscribe();

如果将 subscribe(Collection)assign() 中的集合参数设置为空集合,也可以实现取消订阅,以下三种方式都可以取消订阅:

consumer.unsubscribe();
consumer.subscribe(new ArrayList<String>());
consumer.assign(new ArrayList<TopicPartition>());

如果没有订阅任何主题或分区, 那么再继续执行消费程序的时候会报出IllegalStateException异常:

java.lang.IllegalStateException : Consumer is not subscribed to any topics orassigned any partitions 

 

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值