微服务架构-利用Redis特性进行业务解耦
背景:
接着上篇文章来,上篇文章讲的是如何利用ApplicationContext的事件机制来达到业务解耦,而且这只能作用在单体应用中。在当下这么盛行的微服务架构中,想要再利用此方案做业务解耦是不可能的了,我们也提到,现在比较流行的解决方案是利用消息队列来完成,例如现在流行的RabbitMQ、RocketMQ、ActiveMQ,Kafka。
当然了,我们还可以利用Redis的队列来完成,也是完全没问题的。刚好我自己的阿里云装好了一个redis,我们就直接用Redis来解决吧。Redis提供了生产/消费模式和发布/订阅模式。这里提一下,生产消费模式适合那些一对一的,因为只能一个消费者去消费:例如用户注册了只发短信提示。而我们之前的例子是一对多的,即用户注册了需要发送短信和发送邮件,所以我们会用到发布订阅模式,只要订阅了某个频道,所有订阅者都能收到这频道的消息,然后来对此进行消费。那么开始吧~
说到微服务我们会想到Spring Cloud,可是我们现在是业务解耦,不需要服务之间直接的调用,所以我们直接只使用Spring Boot做微服务架构即可。简单分四大模块,用户模块(hyf-user)、短信模块(hyf-message)、邮件模块(hyf-mail)、公共类模块(hyf-encapsulation)。
开始:
1、项目结构如下图所示:
2、因为使用Redis,所以我们首先得引入Redis相关依赖:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-pool2</artifactId>
</dependency>
3、并且在application.properties写上配置:
# redis服务端口
spring.redis.port=6379
# redis的服务地址
spring.redis.host=127.0.0.1
# 如果redis设置了密码这里也要配上
spring.redis.password=xxx
# 连接超时时间(毫秒)
spring.redis.timeout=10000
# Redis默认情况下有16个分片,这里配置具体使用的分片,默认是0
spring.redis.database=0
# 连接池最大连接数(使用负值表示没有限制) 默认 8
spring.redis.lettuce.pool.max-active=8
# 连接池最大阻塞等待时间(使用负值表示没有限制) 默认 -1
spring.redis.lettuce.pool.max-wait=-1
# 连接池中的最大空闲连接 默认 8
spring.redis.lettuce.pool.max-idle=8
# 连接池中的最小空闲连接 默认 0
spring.redis.lettuce.pool.min-idle=0
4、发布信息:
用户Service直接调用默认的StringRedisTemplate去给指定的通道发布消息即可
/**
* @author Howinfun
* @desc 用户Service
* @date 2019/5/13
*/
@Service
@Slf4j
@AllArgsConstructor
public class UserService {
private StringRedisTemplate stringRedisTemplate;
/**
* 用户注册
* @param user
*/
public void registerUser(User user){
log.info("用户:"+user.getName()+"注册成功");
// 给redis的channel中发布消息
String userInfo = JSON.toJSONString(user);
stringRedisTemplate.convertAndSend(UserConstants.USER_REGISTER,userInfo);
}
}
5、订阅频道:
1、订阅稍微麻烦一点,首先抽象一个消费信息的接口:
public interface AbstractReceiver {
// 消费消息的方法
void receiveMessage(Object message);
}
2、然后发送短信需要创建一个类去实现此接口,然后在重写方法里头实现自己的业务逻辑(发送邮件的同理):
/**
* @author Howinfun
* @desc
* @date 2019/5/14
*/
@Component
public class MessageReceiver implements AbstractReceiver {
@Autowired
private MessageService messageService;
@Override
public void receiveMessage(Object message) {
User user = JSON.parseObject((String) message, User.class);
// 发送短信的业务逻辑
messageService.sendMessage(user);
}
}
3、然后需要给短信订阅弄一个配置类(发送邮件的同理):
/**
* @author Howinfun
* @desc
* @date 2019/5/14
*/
@Configuration
public class RedisConfig {
/**
* redis消息监听器容器
* 可以添加多个监听不同话题的redis监听器,只需要把消息监听器和相应的消息订阅处理器绑定,该消息监听器
* 通过反射技术调用消息订阅处理器的相关方法进行一些业务处理
* @param messageListener
* @return
*/
@Bean
RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory,
MessageListenerAdapter messageListener) {
RedisMessageListenerContainer container = new RedisMessageListenerContainer();
container.setConnectionFactory(connectionFactory);
//mailListener订阅了一个叫user:register 的通道
container.addMessageListener(messageListener, new PatternTopic(UserConstants.USER_REGISTER));
return container;
}
/**
* 消息监听器适配器,绑定消息处理器,利用反射技术调用消息处理器的业务方法
* @param receiver
* @return
*/
@Bean
MessageListenerAdapter messageListener(MessageReceiver receiver) {
//这个地方 是给messageListenerAdapter 传入一个消息接受的处理器,利用反射的方法调用“receiveMessage”
//MessageListenerAdapter提供的默认调用处理器的方法是handleMessage 可以自己到源码里面看
// 所以如果我们定义的方法不是这个,需要在构造函数这添加上
return new MessageListenerAdapter(receiver, "receiveMessage");
}
}
6、最后我们就可以启动项目来测试一下了,可以看到已经成功了:
hyf-user控制台:
2019-05-15 09:55:16.493 INFO 13672 --- [nio-8080-exec-2] com.hyf.user.service.UserService : 用户:howinfun注册成功
2019-05-15 09:55:16.626 INFO 13672 --- [nio-8080-exec-2] io.lettuce.core.EpollProvider : Starting without optional epoll library
2019-05-15 09:55:16.627 INFO 13672 --- [nio-8080-exec-2] io.lettuce.core.KqueueProvider : Starting without optional kqueue library
hyf-message控制台:
2019-05-15 09:55:17.422 INFO 8088 --- [ container-2] com.hyf.message.service.MessageService : 给用户howinfun发送短信,手机号码为:12345678900
hyf-mail控制台:
2019-05-15 09:55:17.422 INFO 15352 --- [ container-2] com.hyf.mail.service.MailService : 给用户howinfun发送邮件,EMail为:baidu@qq.com
最后
因为不想篇幅太长,所以只放上了核心代码,需要详细了解的可到码云上看:Redis解决业务解耦源码
可能有些同学会问到,万一redis挂了呢,那就岂不是没得发动短信和发动邮件了?答案是对的~哈哈哈,挂了那当然就没得发送了。不过对于这种关联性不强的没啥所谓,注册完没法短信也不是特别大的事情,但是呢,如果是电商的项目,下单和减少库存可是要强一致性的,那么有啥方案,可参考https://mp.weixin.qq.com/s/FAlv-qE1jjiiF0JPoMVcWA?
微服务架构-利用Redis特性进行业务解耦的更多相关文章
- 微服务架构中的Redis
了解如何将Redis与Spring Cloud和Spring Data一起使用以提供配置服务器,消息代理和数据库. Redis可以广泛用于微服务架构中.它可能是少数流行的软件解决方案之一,你的应用程序 ...
- 一文读懂 Spring Boot、微服务架构和大数据治理三者之间的故事
微服务架构 微服务的诞生并非偶然,它是在互联网高速发展,技术日新月异的变化以及传统架构无法适应快速变化等多重因素的推动下诞生的产物.互联网时代的产品通常有两类特点:需求变化快和用户群体庞大,在这种情况 ...
- Spring Boot、微服务架构和大数据
一文读懂 Spring Boot.微服务架构和大数据治理三者之间的故事 https://www.cnblogs.com/ityouknow/p/9034377.html 微服务架构 微服务的诞生并非偶 ...
- Re:从 0 开始的微服务架构--(四)如何保障微服务架构下的数据一致性--转
原文地址:http://mp.weixin.qq.com/s/eXvoJew3bjFKzLLJpS0Otg 随着微服务架构的推广,越来越多的公司采用微服务架构来构建自己的业务平台.就像前边的文章说的, ...
- 【Spring Cloud】微服务架构选型方案
1.技术架构 2.组件介绍 1.服务注册与发现——Eureka 服务注册与发现中心采用Eureka,以AP为核心的高可用注册中心,保证高可用性和最终一致性,server之间互相注册的replicate ...
- 网易蜂巢微服务架构:用RabbitMQ实现轻量级通信
本次分享内容由三个部分组成: 微服务架构与MQ RabbitMQ场景分析与优化 RabbitMQ在网易蜂巢中的应用和案例分享 1微服务架构与MQ 微服务架构是一种架构模式,它将单体应用划分成一组微小的 ...
- 微服务架构基础之Service Mesh
ServiceMesh(服务网格) 概念在社区里头非常火,有人提出 2018 年是 ServiceMesh 年,还有人提出 ServiceMesh 是下一代的微服务架构基础. 那么到底什么是 Serv ...
- Scala微服务架构 一
因为公司的一个项目需求变动会非常频繁,同时改动在可控范围内,加上产品同学喜欢一些超前思维,我们的CTO决定提前开启八门,使用微服务架构. 划重点 微服务架构主要特点: ==独立组件(自主开发升级)== ...
- 浅谈微服务架构、容器技术与K8S
关注嘉为科技,获取运维新知 企业应用系统:从单体应用走向微服务架构:从裸金属走向容器. 如果在诸多热门云计算技术诸如容器.微服务.DevOps.OpenStack等之中,找出一个最火的方向,那么可能非 ...
随机推荐
- CVE-2019-0708远程桌面代码执行漏洞复现
漏洞环境 使用VMware 安装Windows7 SP1模拟受害机 利用 攻击工具准备 1.使用如下命令一键更新安装的metasploit框架 curl https://raw.githubuserc ...
- 移动端适配,h5网页,手机端适配兼容方案.可以显示真实的1px边框和12px字体大小,dpr浅析
以前写移动端都是用这段JS解决. (function (doc, win) { // 分辨率Resolution适配 var docEl = doc.documentElement, resizeEv ...
- Docker竟然还能这么玩?商业级4G代理搭建实战!
时间过得真快,距离这个系列的上一篇文章<商业级4G代理搭建指南[准备篇]>发布的时间已经过了两个星期了,上个星期由于各种琐事缠身,周二开始就没空写文章了,所以就咕咕咕了. 那么在准备篇中, ...
- 〈一〉ElasticSearch的介绍
目录 什么是ElasticSearch 核心能力 ES的搜索核心 搜索引擎选择 搜索的处理 补充: 小节总结: 基本学习环境搭建 如何操作ElasticSearch 下载.安装和运行(Based Wi ...
- 将dos格式文件转换为unix格式
在windows下换行符是\r\n,表示回到行首并换到下一行 而unix系统中换行符是\n 这样就存在一个问题,在windows上的文档到了unix上可能就无法使用了 针对这个情况有几种解决办法: 1 ...
- 前端黑魔法:webworker动态化,无需JS文件创建worker
前言 前几天,我和一位知乎网友讨论这个问题的时候,觉得这非常有意思,所以写了这篇文章作为记录 本文的思路和项目代码来源于知友 @simon3000,我加以修饰以更符合理解的需求. 本文所用代码已经 ...
- celery详解
目录 Celery详解 1.背景 2.形象比喻 3.celery具体介绍 3.1 Broker 3.2 Backend 4.使用 4.1 celery架构 4.2 安装redis+celery 4.3 ...
- JS 防抖和节流
防抖和节流 在处理高频事件,类似于window的resize或者scorll,或者input输入校验等操作时.如果直接执行事件处理器,会增大浏览器的负担,严重的直接卡死,用户体验非常不好. 面对这种情 ...
- 集群某节点DataNode服务无法启动解决(报java.net.BindException:Address already in use错误)
现象: 在集群中某节点, 启动DataNode服务后马上又Shutdown, 在操作系统没看到有DataNode的日志(可能是服务启动失败, 自动删除了日志文件),幸好在界面上可以查看报错的日志: ...
- SqlServer Left、Right、CharIndex函数
LEFT 函数:返回字符串中从左边开始指定个数字符 RIGT.H 函数:返回字符串从右边开始指定个数字符 len函数:LEN 函数返回文本字段中值的长度. CHARINDEX函数:CHARINDEX ...