java消息队列怎么异步_使用消息队列扩展异步执行的实现方式

本文介绍了如何利用消息队列替代Spring的@Async注解进行异步任务处理,以解决线程池方式的不足,如监控不便、上下文丢失等问题。通过注解+拦截器的方式,结合RabbitMQ的监听器,实现将任务丢入消息队列,从而在集群中分布压力并确保任务执行的正确性。此外,还展示了如何设置消息队列的Header来保存上下文信息,以及如何配置Json转换器进行对象传输。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

背景

你可能在你的项目中用过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(Queue[] queues) {

String s = this.beanFactory.resolveEmbeddedValue(queues[0].value());

return (String) resolver.evaluate(s, evalContext);

}

rabbitTemplate.convertAndSend(resolveKey(queues), args[0]);

注入消息队列的Json转换器,方便对象传输

@Bean

public Jackson2JsonMessageConverter producerJackson2MessageConverter() {

return new Jackson2JsonMessageConverter();

}

如有需要,我们可以将上下文的用户信息、token等写入消息队列的Header中

private MessagePostProcessor beforePublishPostProcessor() {

return message -> {

// setting up context to message header

return message;

};

}

被异步调用的service代码:

@Service

@Slf4j

public class DemoService {

@RabbitListener(queuesToDeclare = @Queue("mytestqueue"))

public void checkSome(List tagTuple) {

log.warn("check here {}", tagTuple);

}

}

调用service的controller:

@RestControllerpublic classDemoController {

@AutowiredprivateDemoService demoService;

@RequestMapping("check")

publicInteger checkSome() {

ArrayList tagTuple = newArrayList<>();

tagTuple.add("bar");

tagTuple.add("foo");

demoService.checkSome(tagTuple);

return0;

}

}

执行查看效果

17:00:14.584TRACE[AbstractHandlerMapping.java:411]Mapped to org.smop.duplex.sample.DemoController#checkSome()

17:00:21.263WARN [DemoService.java:16]check here [bar, foo]

如有帮助或启发,还请点个👍

附代码仓库:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值