项目启动时自动订阅配置中指定的MQ主题,并将接收到的消息存储在Redis中,提供了一个HTTP接口来查询存储的消息

涵盖从application.yml配置,通过MQ订阅消息,将消息存放到Redis,最后通过HTTP接口提供消息查询的整个流程,我们将分步进行。

Step 1: application.yml配置

yaml 复制代码
spring:
  profiles:
    active: dev

  redis:
    host: localhost
    port: 6379
    database: 0
    jedis:
      pool:
        max-active: 10
        max-idle: 5
        min-idle: 0
        max-wait: -1ms

mq:
  subscribe:
    enable: true
    topics:
      - topic1
      - topic2
    tag: "*"
    consumerGroup: "yourConsumerGroup"

这个配置文件定义了Redis的连接信息和MQ订阅的相关配置,包括是否启动时自动订阅MQ和订阅的主题列表。

Step 2: 消息存储服务(MessageStorage)

java 复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Service;
import java.util.Map;

@Service
public class MessageStorage {
    private final StringRedisTemplate redisTemplate;

    @Autowired
    public MessageStorage(StringRedisTemplate redisTemplate) {
        this.redisTemplate = redisTemplate;
    }

    public void storeMessage(String topic, String messageId, String message) {
        redisTemplate.opsForHash().put("mq:topic:" + topic, messageId, message);
    }

    public String getMessageByTopicAndId(String topic, String messageId) {
        Object message = redisTemplate.opsForHash().get("mq:topic:" + topic, messageId);
        return message != null ? message.toString() : null;
    }

    public Map<Object, Object> getAllMessagesByTopic(String topic) {
        return redisTemplate.opsForHash().entries("mq:topic:" + topic);
    }
}

Step 3: ConsumerManager

java 复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Service;

@Service
public class ConsumerManager implements CommandLineRunner {

    @Value("${mq.subscribe.enable:false}")
    private boolean autoSubscribe;

    @Value("${mq.subscribe.topics}")
    private List<String> topics;

    @Value("${mq.subscribe.tag}")
    private String tag;

    @Value("${mq.subscribe.consumerGroup}")
    private String consumerGroup;

    @Autowired
    private MessageStorage messageStorage;

    // 示例方法,需根据实际MQ客户端进行实现
    public void startConsumer(String topic, String tag, String consumerGroup) {
        // 假设的MQ订阅逻辑
        // 省略实际的MQ订阅代码
        // 假设收到消息后:
        String messageId = "uniqueMessageId"; // 假设的消息ID
        String message = "Example Message"; // 假设的消息内容
        messageStorage.storeMessage(topic, messageId, message);
    }

    @Override
    public void run(String... args) {
        if (autoSubscribe) {
            topics.forEach(topic -> {
                try {
                    startConsumer(topic, tag, consumerGroup);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            });
        }
    }
}

Step 4: ConsumerController(HTTP接口)

java 复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

import java.util.Map;

@RestController
@RequestMapping("/api/messages")
public class ConsumerController {

    @Autowired
    private MessageStorage messageStorage;

    @GetMapping("/get")
    public ResponseEntity<?> getMessage(@RequestParam String topic, @RequestParam(required = false) String messageId) {
        if (messageId != null) {
            String message = messageStorage.getMessageByTopicAndId(topic, messageId);
            return message != null ? ResponseEntity.ok(message) : ResponseEntity.notFound().build();
        } else {
            Map<Object, Object> messages = messageStorage.getAllMessagesByTopic(topic);
            return ResponseEntity.ok(messages);
        }
    }
}

说明

  • MQ订阅逻辑ConsumerManager中的startConsumer方法需要根据你使用的MQ客户端库具体实现,这里仅提供了一个概念性的示例。
  • 消息存储与检索MessageStorage类使用Redis的Hash数据结构

来存储和检索消息。每个topic对应一个Hash,其中每个messageId是键,消息内容是值。

  • HTTP接口ConsumerController提供了一个/api/messages/get接口,允许根据topic和可选的messageId查询消息。

通过上述步骤,我们完成了一个简单的系统,它能够在项目启动时自动订阅配置中指定的MQ主题,并将接收到的消息存储在Redis中。同时,我们也提供了一个HTTP接口来查询存储的消息。这个示例提供了一个基本框架,你可以根据实际的需求和MQ客户端库进行调整和扩展。

相关推荐
大厂技术总监下海18 分钟前
用户行为分析怎么做?ClickHouse + 嵌套数据结构,轻松处理复杂事件
大数据·数据结构·数据库
alonewolf_9927 分钟前
深入理解MySQL事务与锁机制:从原理到实践
android·数据库·mysql
CCPC不拿奖不改名35 分钟前
网络与API:从HTTP协议视角理解网络分层原理+面试习题
开发语言·网络·python·网络协议·学习·http·面试
朝依飞44 分钟前
fastapi+SQLModel + SQLAlchemy2.x+mysql
数据库·mysql·fastapi
3***g2051 小时前
redis连接服务
数据库·redis·bootstrap
m0_598177231 小时前
SQL 方法函数(1)
数据库
oMcLin1 小时前
如何在Oracle Linux 8.4上通过配置Oracle RAC集群,确保企业级数据库的高可用性与负载均衡?
linux·数据库·oracle
信创天地1 小时前
核心系统去 “O” 攻坚:信创数据库迁移的双轨运行与数据一致性保障方案
java·大数据·数据库·金融·架构·政务
tzy2331 小时前
分享一个 HTTP(S) 代理&抓包工具,拦截和Mock Web客户端请求和服务端响应
前端·网络协议·http
胖咕噜的稞达鸭1 小时前
进程间的通信(1)(理解管道特性,匿名命名管道,进程池,systeam V共享内存是什么及优势)重点理解代码!
linux·运维·服务器·数据库