Redis高级(消息队列)
消息队列(Message Queue)#
1.什么是消息#
两台设备(例如服务与服务)之间传递的数据就可以称之为消息
2.什么是消息队列#
消息队列是在消息的传输过程中保存消息的容器。
为什么使用消息队列#
异步
例如发送验证码
解耦
例如服务与服务之间的调用
削峰限流
一个消息队列的基本组成 #
生产者producer
根据主题发送消息
消费者consumer
根据主题接收消息
代理者broker
存储消息,自动分发消息
主题topic
区分消息
消费者监听Listener
监听消息的到来
图示

编辑 Springboot整合redis订阅发布基本流程#
1.定义Topic(主题)/Channel(频道)
使用接口定义Topic常量
2.创建订阅者类,接收发布者发布的消息
实现MessageListener,重写onMessage方法
3.添加订阅者类对象到容器中
注入redis连接工厂RedisConnectionFactory
获取redis连接getConnection
调用订阅方法
subscribe
4.测试发布订阅
convertAndSend
基本使用#
1.在service模块下新建listener包#
创建
package com.zzyl.listener;
import lombok.extern.slf4j.Slf4j;
import org.springframework.data.redis.connection.Message;
import org.springframework.data.redis.connection.MessageListener;
import org.springframework.stereotype.Component;
@Slf4j
@Component
public class RedisQueueListener implements MessageListener {
@Override
public void onMessage(Message message, byte[] pattern) {
String s = new String(message.getBody());
String topic = new String(message.getChannel());
log.info("监听到消息{},来自{}",s,topic);
}
}

2.创建topic#
package com.zzyl.constant;
public interface RedisQueueTopic {
String TEST_TOPIC = "test_topic";
String TEST_TOPIC2 = "test_topic2";
}

3.创建配置类#
package com.zzyl.config;
import com.zzyl.constant.RedisQueueTopic;
import com.zzyl.listener.RedisQueueListener;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnection;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import javax.annotation.Resource;
@Configuration
public class RedisQueueConfig {
@Resource
private RedisQueueListener redisQueueListener;
@Bean
public RedisConnectionFactory redisQueueConnectionFactory(RedisConnectionFactory redisConnectionFactory){
RedisConnection connection = redisConnectionFactory.getConnection();
connection.subscribe(redisQueueListener, RedisQueueTopic.TEST_TOPIC.getBytes(),RedisQueueTopic.TEST_TOPIC2.getBytes());
return redisConnectionFactory;
}
}

4.测试使用#
package com.zzyl;
import com.zzyl.constant.RedisQueueTopic;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.data.redis.core.RedisTemplate;
import javax.annotation.Resource;
@SpringBootTest
public class TestRedisQueue {
@Resource
private RedisTemplate<String,Object> redisTemplate;
@Test
public void test(){
redisTemplate.convertAndSend(RedisQueueTopic.TEST_TOPIC,"hello");
}
@Test
public void tes1t(){
redisTemplate.convertAndSend(RedisQueueTopic.TEST_TOPIC2,"hello2");
}
}

5.输出结果#
2024-09-26 10:38:33.724 [lettuce-nioEventLoop-4-2] INFO com.zzyl.listener.RedisQueueListener-监听到消息"hello2",来自test_topic2

接收Iot平台消息服务 #
1.导入jar包#
<!-- amqp 1.0 qpid client -->
<dependency>
<groupId>org.apache.qpid</groupId>
<artifactId>qpid-jms-client</artifactId>
<version>0.57.0</version>
</dependency>
<!-- util for base64-->
<dependency>
<groupId>commons-codec</groupId>
<artifactId>commons-codec</artifactId>
<version>1.10</version>
</dependency>

2.新建服务#
3.新服务jar包#
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.7.4</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>com.lu</groupId>
<artifactId>zzyl-transfer-server</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>zzyl-transfer-server</name>
<description>zzyl-transfer-server</description>
<url/>
<licenses>
<license/>
</licenses>
<developers>
<developer/>
</developers>
<scm>
<connection/>
<developerConnection/>
<tag/>
<url/>
</scm>
<properties>
<java.version>17</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<!--阿里云iot平台对接服务端订阅依赖-->
<!-- amqp 1.0 qpid client -->
<dependency>
<groupId>org.apache.qpid</groupId>
<artifactId>qpid-jms-client</artifactId>
<version>0.57.0</version>
</dependency>
<!-- util for base64-->
<dependency>
<groupId>commons-codec</groupId>
<artifactId>commons-codec</artifactId>
<version>1.10</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.75</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<excludes>
<exclude>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</exclude>
</excludes>
</configuration>
</plugin>
</plugins>
</build>
</project>

4.redis配置类 #
主要用于连接redis操作redis
package com.lu.zzyltransferserver.config;
import com.alibaba.fastjson.support.spring.GenericFastJsonRedisSerializer;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisStandaloneConfiguration;
import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.serializer.RedisSerializer;
/**
* @author L
*/
@Configuration
public class RedisConfig {
@Value("${spring.redis.host}")
private String host;
@Value("${spring.redis.port}")
private int port;
@Value("${spring.redis.password}")
private String password;
@Value("${spring.redis.database}")
private int database;
@Value("${spring.redis.username}")
private String username;
@Bean
public LettuceConnectionFactory redisConnectionFactory(){
//配置redis数据库连接池信息
RedisStandaloneConfiguration configuration = new RedisStandaloneConfiguration();
configuration.setHostName(host);
configuration.setPort(port);
configuration.setDatabase(database);
configuration.setPassword(password);
configuration.setUsername(username);
LettuceConnectionFactory lettuceConnectionFactory = new LettuceConnectionFactory(configuration);
return lettuceConnectionFactory;
}
@Bean
public RedisTemplate<?,?> redisTemplate(){
//创建redis操作Bean实例对象
RedisTemplate<?,?> stringRedisTemplate = new StringRedisTemplate();
stringRedisTemplate.setConnectionFactory(redisConnectionFactory());
return stringRedisTemplate ;
}
//中转服务使用@Scheduled注解,写定时任务
//每五分钟往redis队列当中生产一条信息,上报老人健康手表设备信息
}

5.配置主题常量类#
RedisQueueTopic
接口:用于定义Redis队列的主题名称,它是一个常量接口。将主题名称定义为接口中的静态常量,方便在项目中引用。
package com.lu.zzyltransferserver.constants;
/**
* @author L
*/
public interface RedisQueueTopic {
String TEST_TOPIC = "test_topic";
String TEST_TOPIC2 = "test_topic2";
}

6.监听转发#
监听阿里云消息,转发消息
package com.lu.zzyltransferserver.listener;
import com.lu.zzyltransferserver.constants.RedisQueueTopic;
import lombok.extern.slf4j.Slf4j;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import javax.jms.Message;
import javax.jms.MessageListener;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
/**
* @author L
*/
@Slf4j
@Component
public class IotListener implements MessageListener {
@Resource
private RedisTemplate<String,Object> redisTemplate;
//业务处理异步线程池,线程池参数可以根据您的业务特点调整,或者您也可以用其他异步方式处理接收到的消息。
private final static ExecutorService executorService = new ThreadPoolExecutor(
Runtime.getRuntime().availableProcessors(),
Runtime.getRuntime().availableProcessors() * 2, 60, TimeUnit.SECONDS,
new LinkedBlockingQueue(50000));
@Override
public void onMessage(Message message) {
try {
//1.收到消息之后一定要ACK。
// 推荐做法:创建Session选择Session.AUTO_ACKNOWLEDGE,这里会自动ACK。
// 其他做法:创建Session选择Session.CLIENT_ACKNOWLEDGE,这里一定要调message.acknowledge()来ACK。
// message.acknowledge();
//2.建议异步处理收到的消息,确保onMessage函数里没有耗时逻辑。
// 如果业务处理耗时过程过长阻塞住线程,可能会影响SDK收到消息后的正常回调。
executorService.submit(new Runnable() {
@Override
public void run() {
processMessage(message);
}
});
} catch (Exception e) {
log.error("submit task occurs exception ", e);
}
}
/**
* 在这里处理您收到消息后的具体业务逻辑。
*/
private void processMessage(Message message) {
try {
byte[] body = message.getBody(byte[].class);
String content = new String(body);
String topic = message.getStringProperty("topic");
String messageId = message.getStringProperty("messageId");
log.info("接收消息"
+ ",\n 来自于 = " + topic
+ ",\n 消息Id = " + messageId
+ ",\n content = " + content);
//作为iot消费者获取iot平台订阅的消息
//作为redis生产者将消息发送到redis的队列中
redisTemplate.convertAndSend(RedisQueueTopic.TEST_TOPIC,content);
} catch (Exception e) {
log.error("processMessage occurs error ", e);
}
}
}

7.与阿里云 IoT 平台进行通信的客户端#
这个代码片段实现了基于 AMQP 协议与阿里云 IoT 平台进行通信的客户端,并且它在 Spring Boot 启动时自动运行,用于与 IoT 平台进行消息的消费。消息消费完成后,会通过 Spring 的
IotListener
来处理消息,接下来通过 Redis 发布订阅将消息传递到其他服务。
package com.lu.zzyltransferserver.runner;
import com.lu.zzyltransferserver.listener.IotListener;
import org.apache.commons.codec.binary.Base64;
import org.apache.qpid.jms.JmsConnection;
import org.apache.qpid.jms.JmsConnectionListener;
import org.apache.qpid.jms.message.JmsInboundMessageDispatch;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.CommandLineRunner;
import org.springframework.stereotype.Component;
import javax.annotation.Resource;
import javax.crypto.Mac;
import javax.crypto.spec.SecretKeySpec;
import javax.jms.*;
import javax.naming.Context;
import javax.naming.InitialContext;
import java.net.InetAddress;
import java.net.URI;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.Hashtable;
import java.util.List;
@Component
public class AmqpClientRunner implements CommandLineRunner {
private final static Logger logger = LoggerFactory.getLogger(AmqpClientRunner.class);
/**
* 工程代码泄露可能会导致 AccessKey 泄露,并威胁账号下所有资源的安全性。以下代码示例使用环境变量获取 AccessKey 的方式进行调用,仅供参考
*/
@Value(value = "${zzyl.aliyun.accessKeyId}")
String accessKey;
@Value(value = "${zzyl.aliyun.accessKeySecret}")
String accessSecret;
@Value(value = "${zzyl.aliyun.consumerGroupId}")
String consumerGroupId;
//iotInstanceId:实例ID。若是2021年07月30日之前(不含当日)开通的公共实例,请填空字符串。
@Value(value = "${zzyl.aliyun.iotInstanceId}")
String iotInstanceId;
@Resource
IotListener iotListener;
//控制台服务端订阅中消费组状态页客户端ID一栏将显示clientId参数。
//建议使用机器UUID、MAC地址、IP等唯一标识等作为clientId。便于您区分识别不同的客户端。
private static String clientId;
static {
try {
clientId = InetAddress.getLocalHost().getHostAddress();
} catch (UnknownHostException e) {
e.printStackTrace();
}
}
//${YourHost}为接入域名,请参见AMQP客户端接入说明文档。
private static String host = "iot-06z00ix0kvl4g9z.amqp.iothub.aliyuncs.com";
// 指定单个进程启动的连接数
// 单个连接消费速率有限,请参考使用限制,最大64个连接
// 连接数和消费速率及rebalance相关,建议每500QPS增加一个连接
private static int connectionCount = 4;
private static JmsConnectionListener myJmsConnectionListener = new JmsConnectionListener() {
/**
* 连接成功建立。
*/
@Override
public void onConnectionEstablished(URI remoteURI) {
logger.info("连接成功建立, 远程uri:{}", remoteURI);
}
/**
* 尝试过最大重试次数之后,最终连接失败。
*/
@Override
public void onConnectionFailure(Throwable error) {
logger.error("连接失败: {}", error.getMessage());
}
/**
* 连接中断。
*/
@Override
public void onConnectionInterrupted(URI remoteURI) {
logger.info("连接中断, 远程uri:{}", remoteURI);
}
/**
* 连接中断后又自动重连上。
*/
@Override
public void onConnectionRestored(URI remoteURI) {
logger.info("连接中断后又自动重连上, 远程uri:{}", remoteURI);
}
@Override
public void onInboundMessage(JmsInboundMessageDispatch envelope) {
}
@Override
public void onSessionClosed(Session session, Throwable cause) {
}
@Override
public void onConsumerClosed(MessageConsumer consumer, Throwable cause) {
}
@Override
public void onProducerClosed(MessageProducer producer, Throwable cause) {
}
};
/**
* 计算签名,password组装方法,请参见AMQP客户端接入说明文档。
*/
private static String doSign(String toSignString, String secret, String signMethod) throws Exception {
SecretKeySpec signingKey = new SecretKeySpec(secret.getBytes(), signMethod);
Mac mac = Mac.getInstance(signMethod);
mac.init(signingKey);
byte[] rawHmac = mac.doFinal(toSignString.getBytes());
return Base64.encodeBase64String(rawHmac);
}
@Override
public void run(String... args) throws Exception {
List<Connection> connections = new ArrayList<>();
//参数说明,请参见AMQP客户端接入说明文档。
for (int i = 0; i < connectionCount; i++) {
long timeStamp = System.currentTimeMillis();
//签名方法:支持hmacmd5、hmacsha1和hmacsha256。
String signMethod = "hmacsha1";
//userName组装方法,请参见AMQP客户端接入说明文档。
String userName = clientId + "-" + i + "|authMode=aksign"
+ ",signMethod=" + signMethod
+ ",timestamp=" + timeStamp
+ ",authId=" + accessKey
+ ",iotInstanceId=" + iotInstanceId
+ ",consumerGroupId=" + consumerGroupId
+ "|";
//计算签名,password组装方法,请参见AMQP客户端接入说明文档。
String signContent = "authId=" + accessKey + "×tamp=" + timeStamp;
String password = doSign(signContent, accessSecret, signMethod);
String connectionUrl = "failover:(amqps://" + host + ":5671?amqp.idleTimeout=80000)"
+ "?failover.reconnectDelay=30";
Hashtable<String, String> hashtable = new Hashtable<>();
hashtable.put("connectionfactory.SBCF", connectionUrl);
hashtable.put("queue.QUEUE", "default");
hashtable.put(Context.INITIAL_CONTEXT_FACTORY, "org.apache.qpid.jms.jndi.JmsInitialContextFactory");
Context context = new InitialContext(hashtable);
ConnectionFactory cf = (ConnectionFactory) context.lookup("SBCF");
Destination queue = (Destination) context.lookup("QUEUE");
// 创建连接。
Connection connection = cf.createConnection(userName, password);
connections.add(connection);
((JmsConnection) connection).addConnectionListener(myJmsConnectionListener);
// 创建会话。
// Session.CLIENT_ACKNOWLEDGE: 收到消息后,需要手动调用message.acknowledge()。
// Session.AUTO_ACKNOWLEDGE: SDK自动ACK(推荐)。
Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
connection.start();
// 创建Receiver连接。
MessageConsumer consumer = session.createConsumer(queue);
//添加iot平台服务端消费者监听
consumer.setMessageListener(iotListener);
}
logger.info("连接成功!");
// 结束程序运行
Thread.sleep(6000 * 1000);
logger.info("run shutdown");
connections.forEach(c -> {
try {
c.close();
} catch (JMSException e) {
logger.error("failed to close connection", e);
}
});
}
}

8.配置yml#
zzyl:
aliyun:
accessKeyId: LTAI5tMA5wSWRmeuyYPei9Vk
accessKeySecret: GSNxohmhEcFa4g7p61RRTzRdABMzcH
consumerGroupId: DEFAULT_GROUP
regionId: cn-shanghai
iotInstanceId: iot-06z00cxspzjxue9
host: iot-06z00cxspzjxue9.amqp.iothub.aliyuncs.com
spring:
redis:
host: 192.168.200.146
port: 6379
password: kK6/fG8&pN7< #指定密码
database: 1 #指定使用数据库
username: default # 用户名

9.业务服务消息监听#
需要要导入redis包
package com.zzyl.listener;
import com.alibaba.fastjson.JSONObject;
import com.zzyl.dto.ContentDTO;
import lombok.extern.slf4j.Slf4j;
import org.springframework.data.redis.connection.Message;
import org.springframework.data.redis.connection.MessageListener;
import org.springframework.stereotype.Component;
@Slf4j
@Component
public class RedisQueueListener implements MessageListener {
@Override
public void onMessage(Message message, byte[] pattern) {
String s = new String(message.getBody());
String topic = new String(message.getChannel());
log.info("监听到消息{},来自{}",s,topic);
ContentDTO contentDTO = JSONObject.parseObject(s, ContentDTO.class);
log.info("设备json字符串转对象:{}",contentDTO);
}
}

10.测试#
测试流程
1.设备上报信息
这里没有设备用MQTTX模拟设备
2.中间件服务收到上报消息
11.业务服务消息订阅者配置#
package com.zzyl.config;
import com.zzyl.constant.RedisQueueTopic;
import com.zzyl.listener.RedisQueueListener;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.RedisConnection;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import javax.annotation.Resource;
@Configuration
public class RedisQueueConfig {
@Resource
private RedisQueueListener redisQueueListener;
@Bean
public RedisConnectionFactory redisQueueConnectionFactory(RedisConnectionFactory redisConnectionFactory){
RedisConnection connection = redisConnectionFactory.getConnection();
connection.subscribe(redisQueueListener, RedisQueueTopic.TEST_TOPIC.getBytes(),RedisQueueTopic.TEST_TOPIC2.getBytes());
return redisConnectionFactory;
}
}

【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· Obsidian + DeepSeek:免费 AI 助力你的知识管理,让你的笔记飞起来!
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了