背景
你可能在你的项目中用过Spring的@Async注解,以此来将部分方法进行异步执行,以此来提高请求的相应效率
但在服务架构不断的演进之中,这种丢入线程池处理的方式带来的缺陷也很明显:
不利于监控
不易保存现场,如果意外停机,处理繁琐
在集群中的某个节点要处理大量异步任务时,无法将压力分担到集群中其他节点
需要注意使用ThreadLocal特性的模块或第三方组件,需要注意上下文丢失的问题
思路
使用消息队列作为异步任务的实现方式,这样我们就可以:
成熟的MQ中间件都提供可视化管理平台
可以用消息队列Header来保存上下文,如用户信息、token等
消息队列的发布-订阅模式可以最大程度利用集群的业务处理能力
更容易保证任务的顺序性
如果有服务节点宕机,可以利用消息确认、消息重试等保证任务执行的正确性
实现
为了保证业务代码和实现方案解耦,类似于@Aync方案,我们同样适用注解+拦截器的方式进行逻辑注入
@Around("@annotation(org.springframework.amqp.rabbit.annotation.RabbitListener)")
public Object cut(ProceedingJoinPoint pjp) throws Throwable {
...
}
实现思路大同小异,就是读取注解中的队列声明,然后以丢入消息队列来替换丢入线程池
private String resolveKey(Que