Redis 基本上是互联网公司必备的工具了,Redis的应用场景实在太多了,但是有很多相似的功能如果每个项目都要实现一遍就显得太麻烦了,所以为了方便,我打算开发一个基于 Redis 的工具集,尽量做到开箱即用。
基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
-
项目地址:https://github.com/YunaiV/ruoyi-vue-pro
-
视频教程:https://doc.iocoder.cn/video/
这个工具集并没有开发完成,实现了部分功能,如下图
简单介绍下已经实现的模块:
-
common : 整个项目公共模块,比如AOP工具等;
-
-
-
-
-
以上的这些模块都是已经实现的了,还有 社交、限流、幂等相关功能后面会陆续实现。
基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
-
项目地址:https://github.com/YunaiV/yudao-cloud
-
视频教程:https://doc.iocoder.cn/video/
1.引入 Maven 依赖
目前可以下载代码上传到自己的私服或者本地仓库,后面会推到 Maven 中央仓库
<dependency>
<groupId>cn.org.wangchangjiugroupId>
<artifactId>redis-util-spring-boot-starterartifactId>
<version>1.0.0-SNAPSHOTversion>
dependency>
2.配置文件(application.yaml)开启各模块功能开关
redis:
util:
mq:
enable: true
delay:
enable: true
3.实现消息发送者
MQ消息发送:
延迟消息发送:
4.实现消息监听器
MQ消息监听器:
延迟消息监听器:
容器启动时,简单来说就是通过springboot自动装配,创建一些Bean,如下图:
值得注意的是,springboot3.X 自动装配方式有点变化,需要创建文件
META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
文件,文件内容就直接写 自动配置类
RedisUtilAutoConfiguration 主自动装配类会 import 各个模块的自动装配类:
我们以 RedisStreamAutoConfiguration 为例:
该装配类生效需要显示打开,然后就是创建各种Bean。
最主要的Bean有:
-
RedisMessageConsumerManager:
该Bean实现了 BeanPostProcessor 接口,主要作用是,获取被注解 RedisMessageListener 修饰的方法,把信息封装在 RedisMessageConsumerContainer 对象里,方便后面反射调用。
-
StreamMessageListenerContainer:
这个Bean主要是做 redis MQ 的配置,比如配置:一次最多获取多少条消息、没有消息时阻塞时间、执行任务的executor、错误处理器、以及消费组、是否自动ACK等配置,具体代码如下:
@Bean(initMethod = "start", destroyMethod = "stop")
@DependsOn("redisMessageConsumerManager")
@ConditionalOnMissingBean
public StreamMessageListenerContainer> streamMessageListenerContainer(@Autowired RedisMessageConsumerManager redisMessageConsumerManager,
@Autowired RedisConnectionFactory redisConnectionFactory,
@Autowired ErrorHandler errorHandler) {
MyRedisStreamProperties.Options options = myRedisStreamProperties.getOptions();
StreamMessageListenerContainer.StreamMessageListenerContainerOptions> containerOptions =
StreamMessageListenerContainer.StreamMessageListenerContainerOptions
.builder()
// 一次最多获取多少条消息
.batchSize(options.getBatchSize())
// 运行 Stream 的 poll task
.executor(getStreamMessageListenerExecutor())
// Stream 中没有消息时,阻塞多长时间,需要比 `spring.redis.timeout` 的时间小
.pollTimeout(options.getPollTimeout())
// 获取消息的过程或获取到消息给具体的消息者处理的过程中,发生了异常的处理
.errorHandler(errorHandler)
.build();
StreamMessageListenerContainer> streamMessageListenerContainer =
StreamMessageListenerContainer.create(redisConnectionFactory, containerOptions);
// 获取 被 RedisMessageListener 注解修饰的 bean
Map consumerContainerGroups =
redisMessageConsumerManager.getConsumerContainerGroups();
// 循环遍历,创建 消费组
consumerContainerGroups.forEach((groupQueue, redisMessageConsumerContainer) -> {
String[] groupQueues = groupQueue.split("#");
// 创建消费组
createGroups(groupQueues);
RedisMessageListener redisMessageListener = redisMessageConsumerContainer.getRedisMessageListener();
if(!redisMessageListener.useGroup()){
// 独立消费 不使用组
streamMessageListenerContainer.receive(StreamOffset.fromStart(groupQueues[1]), new DefaultGroupStreamListener(redisMessageConsumerContainer));
} else {
// 消费组 消费
if(redisMessageListener.autoAck()){
// 自动ACK
streamMessageListenerContainer.receiveAutoAck(Consumer.from(groupQueues[0], "consumer:" + UUID.randomUUID()),
StreamOffset.create(groupQueues[1], ReadOffset.lastConsumed()), new DefaultGroupStreamListener(redisMessageConsumerContainer));
} else {
// 手动 ACK
streamMessageListenerContainer.receive(Consumer.from(groupQueues[0], "consumer:" + UUID.randomUUID()),
StreamOffset.create(groupQueues[1], ReadOffset.lastConsumed()), new DefaultGroupStreamListener(redisMessageConsumerContainer));
}
}
});
return streamMessageListenerContainer;
}
/**