我的编程空间,编程开发者的网络收藏夹
学习永远不晚

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

今天小编给大家分享一下springboot+kafka中@KafkaListener动态指定多个topic怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

说明

本项目为springboot+kafak的整合项目,故其用了springboot中对kafak的消费注解@KafkaListener

首先,application.properties中配置用逗号隔开的多个topic。

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

方法:利用Spring的SpEl表达式,将topics 配置为:@KafkaListener(topics = “#{’${topics}’.split(’,’)}”)

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

运行程序,console打印的效果如下:

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

因为只开了一条消费者线程,所以所有的topic和分区都分配给这条线程。

如果你想开多条消费者线程去消费这些topic,添加@KafkaListener注解的参数concurrency的值为自己想要的消费者个数即可(注意,消费者数要小于等于你开的所有topic的分区数总和)

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

运行程序,console打印的效果如下:

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

总结一下大家问的最多的一个问题

如何在程序运行的过程中,改变topic,消费者能够消费修改后的topic?

ans: 经过尝试,使用@KafkaListener注解实现不了此需求,在程序启动的时候,程序就会根据@KafkaListener的注解信息初始化好消费者去消费指定好的topic。如果在程序运行的过程中,修改topic,不会让此消费者修改消费者的配置再重新订阅topic的。

不过我们可以有个折中的办法,就是利用@KafkaListener的topicPattern参数来进行topic匹配。

终极方法

思路

不使用@KafkaListener,使用kafka原生客户端依赖,手动初始化消费者,开启消费者线程。

在消费者线程中,每次循环都从配置、数据库或者其他配置源获取最新的topic信息,与之前的topic比较,如果发生变化,重新订阅topic或者初始化消费者。

实现

加入kafka客户端依赖(本次测试服务端kafka版本:2.12-2.4.0)

<dependency><groupId>org.apache.kafka</groupId><artifactId>kafka-clients</artifactId><version>2.3.0</version></dependency>

代码

@Service@Slf4jpublic class KafkaConsumers implements InitializingBean {        private static KafkaConsumer<String, String> consumer;        private List<String> topicList;    public static String getNewTopic() {        try {            return org.apache.commons.io.FileUtils.readLines(new File("D:/topic.txt"), "utf-8").get(0);        } catch (IOException e) {            e.printStackTrace();        }        return null;    }        public KafkaConsumer<String, String> getInitConsumer(List<String> topicList) {        //配置信息        Properties props = new Properties();        //kafka服务器地址        props.put("bootstrap.servers", "192.168.9.185:9092");        //必须指定消费者组        props.put("group.id", "haha");        //设置数据key和value的序列化处理类        props.put("key.deserializer", StringDeserializer.class);        props.put("value.deserializer", StringDeserializer.class);        //创建消息者实例        KafkaConsumer<String, String> consumer = new KafkaConsumer<>(props);        //订阅topic的消息        consumer.subscribe(topicList);        return consumer;    }        @Override    public void afterPropertiesSet() {        // 初始化topic        topicList = Splitter.on(",").splitToList(Objects.requireNonNull(getNewTopic()));        if (org.apache.commons.collections.CollectionUtils.isNotEmpty(topicList)) {            consumer = getInitConsumer(topicList);            // 开启一个消费者线程            new Thread(() -> {                while (true) {                    // 模拟从配置源中获取最新的topic(字符串,逗号隔开)                    final List<String> newTopic = Splitter.on(",").splitToList(Objects.requireNonNull(getNewTopic()));                    // 如果topic发生变化                    if (!topicList.equals(newTopic)) {                        log.info("topic 发生变化:newTopic:{},oldTopic:{}-------------------------", newTopic, topicList);                        // method one:重新订阅topic:                        topicList = newTopic;                        consumer.subscribe(newTopic);                        // method two:关闭原来的消费者,重新初始化一个消费者                        //consumer.close();                        //topicList = newTopic;                        //consumer = getInitConsumer(newTopic);                        continue;                    }                    ConsumerRecords<String, String> records = consumer.poll(Duration.ofMillis(100));                    for (ConsumerRecord<String, String> record : records) {                        System.out.println("key:" + record.key() + "" + ",value:" + record.value());                    }                }            }).start();        }    }}

说一下第72行代码:

ConsumerRecords<String, String> records = consumer.poll(Duration.ofMillis(100));

上面这行代码表示:在100ms内等待Kafka的broker返回数据.超市参数指定poll在多久之后可以返回,不管有没有可用的数据都要返回。

在修改topic后,必须等到此次poll拉取的消息处理完,while(true)循环的时候检测topic发生变化,才能重新订阅topic.

poll()方法一次拉取得消息数默认为:500,如下图,kafka客户端源码中设置的。

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

如果想自定义此配置,可在初始化消费者时加入

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

运行结果(测试的topic中都无数据)

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

注意:KafkaConsumer是线程不安全的,不要用一个KafkaConsumer实例开启多个消费者,要开启多个消费者,需要new 多个KafkaConsumer实例。

以上就是“springboot+kafka中@KafkaListener动态指定多个topic怎么实现”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

springboot+kafka中@KafkaListener动态指定多个topic怎么实现

今天小编给大家分享一下springboot+kafka中@KafkaListener动态指定多个topic怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后
2023-07-04

springboot+kafka中@KafkaListener动态指定多个topic问题

这篇文章主要介绍了springboot+kafka中@KafkaListener动态指定多个topic问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2022-12-27

springBoot项目中怎么实现启动多个实例

今天就跟大家聊聊有关springBoot项目中怎么实现启动多个实例,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。1) 这是不同环境的yml文件,注意 文件名 必须为applicati
2023-06-20

怎么在Java项目中实现一个动态与静态绑定

本篇文章为大家展示了怎么在Java项目中实现一个动态与静态绑定,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Java的动态绑定所谓的动态绑定就是指程执行期间(而不是在编译期间)判断所引用对象的实际类
2023-05-31

dubbo怎么实现consumer从多个group中调用指定group的provider

本篇内容主要讲解“dubbo怎么实现consumer从多个group中调用指定group的provider”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“dubbo怎么实现consumer从多个g
2023-07-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录