当前位置:网站首页>采用注解+拦截器的方式进行异步执行的实现方式
采用注解+拦截器的方式进行异步执行的实现方式
2020-11-08 20:18:00 【osc_03803522】
背景
你可能在你的项目中用过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<String> tagTuple) {
log.warn("check here {}", tagTuple);
}
}
调用service的controller:
@RestController
public class DemoController {
@Autowired
private DemoService demoService;
@RequestMapping("check")
public Integer checkSome() {
ArrayList<String> tagTuple = new ArrayList<>();
tagTuple.add("bar");
tagTuple.add("foo");
demoService.checkSome(tagTuple);
return 0;
}
}
执行查看效果
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]
外部文件操作django的models
#外部文件使用django的models,需要配置django环境
import os
if __name__ == '__main__':
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "singletablehw.settings")
import django
django.setup()
from app01 import models
import datetime
obj_list = []
for i in range(1,10):
obj = models.Book(
title='葵花宝典第%s式'%i,
price=20 + i,
pub_date='198%s-11-11 00:00:00'%i,
# pub_date=datetime.datetime.now(),
publish= '吴老板出版社' if i < 5 else '太白出版社',
)
obj_list.append(obj)
models.Book.objects.bulk_create(obj_list)
url别名反向解析
#添加书籍
url(r'^add_book/', views.add_book,name='abook'), #name='abook' 别名
# 删除书籍
url(r'^delete_book/(\d+)/', views.delele_book,name='delete_book'),
视图:
from django.urls import reverse
reverse('别名') reverse('abook') -- /add_book/ #不带参数的
print(reverse('delete_book',args=(71,))) #/delete_book/71/ 带参数的
模板:
{% url 'abook' %} 无参数的
{% url 'delete_book' book.id %} 有参数的
版权声明
本文为[osc_03803522]所创,转载请带上原文链接,感谢
https://my.oschina.net/u/4797210/blog/4708314
边栏推荐
- Django's simple user system (3)
- 我用 Python 找出了删除我微信的所有人并将他们自动化删除了
- Flink's sink: a preliminary study
- 给大家介绍下,这是我的流程图软件 —— draw.io
- 都说程序员钱多空少,程序员真的忙到没时间回信息了吗?
- 构造回文的最小插入次数
- C/C++学习日记:原码、反码和补码
- Array acquaintance
- How much faster is a server equipped with a SSD than a mechanical hard disk
- Awk implements SQL like join operation
猜你喜欢
第二章编程练习
[elastic search technology sharing] - ten pictures to show you the principle of ES! Understand why to say: ES is quasi real time!
Solution to cross domain problem of front end separation
Flink series (0) -- Preparation (basic stream processing)
Solve the failure of go get download package
【Elasticsearch 技术分享】—— 十张图带大家看懂 ES 原理 !明白为什么说:ES 是准实时的!
Introduction to latex
Dynamic programming: maximum subarray
接口测试工具Eolinker进行post请求
实验
随机推荐
Server side resolution of lengthfieldbasedframedecoder of GetBytes
接口测试工具Eolinker进行post请求
第一部分——第2章指针操作
第一部分——第1章概述
Dynamic planning
C + + opencv4.3 sift matching
都说程序员钱多空少,程序员真的忙到没时间回信息了吗?
If the programming language as martial arts unique! C++ is Jiu Yin Jing. What about programmers?
选择排序
How much faster is a server equipped with a SSD than a mechanical hard disk
IT行业薪资一直遥遥领先!十年后的程序员,是否还是一个高薪职业?
C/C++学习日记:原码、反码和补码
Countdownlatch explodes instantly! Based on AQS, why can cyclicbarrier be so popular?
简明 VIM 练级攻略
npm install 无响应解决方案
Jsliang job series - 07 - promise
getBytes之 LengthFieldBasedFrameDecoder服务端解析
.NET Core 跨平台资源监控库及 dotnet tool 小工具
AI perfume is coming. Will you buy it?
Solution to cross domain problem of front end separation