当前位置:   article > 正文

kafka:java集成 kafka(springboot集成、客户端集成)_java集成kafka

java集成kafka

摘要

对于java的kafka集成,一般选用springboot集成kafka,但可能由于对接方kafka老旧、kafka不安全等问题导致kafak版本与spring版本不兼容,这个时候就得自己根据kafka客户端api集成了。

一、springboot集成kafka

具体官方文档地址:https://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/

1、加入依赖spring-boot-starter-web和spring-kafka 的版本号可以看它们依赖的spring版本是否一致,这里pom依赖如下:

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-web</artifactId>
  4. <version>2.7.9</version>
  5. </dependency>
  6. <dependency>
  7. <groupId>org.springframework.kafka</groupId>
  8. <artifactId>spring-kafka</artifactId>
  9. <version>2.9.6</version>
  10. </dependency>

2、添加application.yml配置,具体如下:

  1. server:
  2. port: 8087
  3. spring:
  4. mvc:
  5. pathmatch:
  6. matching-strategy: ant_path_matcher
  7. kafka:
  8. bootstrap-servers: 192.168.189.128:9092,92.168.189.128:9093,192.168.189.128:9094
  9. consumer:
  10. properties:
  11. group:
  12. id: boot-kafka

3、发送消息,由于KafkaTemplate是自动装配的,所以只要在spring的bean里注入KafkaTemplate发送消息即可,具体如下:

  1. package com.longqi.bootkafka.controller;
  2. import com.longqi.bootkafka.entity.MessageParam;
  3. import com.longqi.bootkafka.entity.Wrapper;
  4. import io.swagger.annotations.Api;
  5. import io.swagger.annotations.ApiOperation;
  6. import io.swagger.annotations.ApiParam;
  7. import lombok.extern.slf4j.Slf4j;
  8. import org.springframework.beans.factory.annotation.Autowired;
  9. import org.springframework.kafka.core.KafkaTemplate;
  10. import org.springframework.web.bind.annotation.PostMapping;
  11. import org.springframework.web.bind.annotation.RequestBody;
  12. import org.springframework.web.bind.annotation.RequestMapping;
  13. import org.springframework.web.bind.annotation.RestController;
  14. import javax.validation.Valid;
  15. /**
  16. * <p>
  17. * 测试 前端控制器
  18. * </p>
  19. * @author LongQi
  20. * @since 2021-06-23
  21. */
  22. @Slf4j
  23. @RestController
  24. @RequestMapping("/test")
  25. @Api(value = "TestController", tags = {"测试 API"})
  26. public class TestController {
  27. @Autowired
  28. private KafkaTemplate<String, String> kafkaTemplate;
  29. private Boolean isSend = true;
  30. @PostMapping("/kafka/sendMessage")
  31. @ApiOperation(httpMethod = "POST", value = "发送kafka告警消息", response = Wrapper.class)
  32. public Wrapper sendKafkaMessage(@Valid @ApiParam("参数") @RequestBody MessageParam param) {
  33. kafkaTemplate.send(param.getTopic(), param.getMessage());
  34. return Wrapper.ok(true);
  35. }
  36. }

这里用参数{"message": "asd54a6d46a4ds","topic": "device-alarm-test"}进行测试,会报如下日志:

发现会报警告:[Producer clientId=producer-1] Error while fetching metadata with correlation id 34 : {device-alarm-test=LEADER_NOT_AVAILABLE},获取主题元数据错误,这个可以忽略,查找元数据失败,kafka默认会自动创建主题的,后续再次发送消息,是不会报这个错误的。

查看可视化工具EFAK,发现主题device-alarm-test是自动创建成功,分区数是kafka的集群配置service.properties里配置的分区9,具体如下:

可以看到,其中一个分区保存了这个消息,logsize变成了1,说明这个消息是发送成功的。另外也可以看到主题的各分区主备消息所在的节点是不一样的。

4、接收消息,接收消息也很简单,只要在spring的bean里使用KafkaListener注解即可,具体如下:

可视化工具也能看到该主题该消费者9个分区的消费情况,具体如下:

logSize为存入分区parttion消息数量,Offset为消费的偏移量(已消费的数量),Lag为未消费的数量(积压的数量),Owner为消费者,目前可以看到消费者为同一个,即只有1个线程在消费这9个分区的消息。

二、客户端集成kafka

直接使用kafka客户端,建议使用最新版的客户端,毕竟没有其他框架版本限制,能用最新的就用最新的,毕竟新的一般性能强也修复了bug。好比23年2月份出现的kafka安全漏洞:远程代码执行漏洞CVE-2023-25194,对现在最新版3.4.0无效,对以前大部分版本就有效。

1、添加依赖,具体如下:

  1. <dependency>
  2. <groupId>org.apache.kafka</groupId>
  3. <artifactId>kafka-clients</artifactId>
  4. <version>3.4.0</version>
  5. </dependency>

2、发送和消费消息,具体代码如下:

  1. package com.longqi.bootkafka.config;
  2. import org.apache.kafka.clients.consumer.ConsumerConfig;
  3. import org.apache.kafka.clients.consumer.ConsumerRecord;
  4. import org.apache.kafka.clients.consumer.ConsumerRecords;
  5. import org.apache.kafka.clients.consumer.KafkaConsumer;
  6. import org.apache.kafka.clients.producer.KafkaProducer;
  7. import org.apache.kafka.clients.producer.ProducerConfig;
  8. import org.apache.kafka.clients.producer.ProducerRecord;
  9. import java.time.Duration;
  10. import java.util.Arrays;
  11. import java.util.Properties;
  12. /**
  13. * @author LongQi
  14. * @projectName boot-integration
  15. * @description: kafka配置
  16. * @date 2023/3/13 14:42
  17. */
  18. public class KafkaConfig {
  19. public static void main(String[] args) {
  20. // 声明主题
  21. String topic = "device-alarm-test";
  22. // 创建消费者
  23. Properties consumerConfig = new Properties();
  24. consumerConfig.put(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, "192.168.189.128:9092,92.168.189.128:9093,192.168.189.128:9094");
  25. consumerConfig.put(ConsumerConfig.GROUP_ID_CONFIG,"boot-kafka");
  26. consumerConfig.put(ConsumerConfig.KEY_DESERIALIZER_CLASS_CONFIG,"org.apache.kafka.common.serialization.StringDeserializer");
  27. consumerConfig.put(ConsumerConfig.VALUE_DESERIALIZER_CLASS_CONFIG,"org.apache.kafka.common.serialization.StringDeserializer");
  28. KafkaConsumer kafkaConsumer = new KafkaConsumer(consumerConfig);
  29. // 订阅主题并循环拉取消息
  30. kafkaConsumer.subscribe(Arrays.asList(topic));
  31. new Thread(new Runnable() {
  32. @Override
  33. public void run() {
  34. while (true){
  35. ConsumerRecords<String, String> records = kafkaConsumer.poll(Duration.ofMillis(10000));
  36. for(ConsumerRecord<String, String> record:records){
  37. System.out.println(record.value());
  38. }
  39. }
  40. }
  41. }).start();
  42. // 创建生产者
  43. Properties producerConfig = new Properties();
  44. producerConfig.put(ProducerConfig.BOOTSTRAP_SERVERS_CONFIG, "192.168.189.128:9092,92.168.189.128:9093,192.168.189.128:9094");
  45. producerConfig.put(ProducerConfig.CLIENT_ID_CONFIG,"boot-kafka-client");
  46. producerConfig.put(ProducerConfig.KEY_SERIALIZER_CLASS_CONFIG, "org.apache.kafka.common.serialization.StringSerializer");
  47. producerConfig.put(ProducerConfig.VALUE_SERIALIZER_CLASS_CONFIG, "org.apache.kafka.common.serialization.StringSerializer");
  48. KafkaProducer producer = new KafkaProducer<>(producerConfig);
  49. // 给主题发送消息
  50. producer.send(new ProducerRecord<>(topic, "hello,"+System.currentTimeMillis()));
  51. }
  52. }

最后可以看到打印消息如下:

成功接收到消息并打印

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小舞很执着/article/detail/937251
推荐阅读
相关标签
  

闽ICP备14008679号