SpringBoot异步调用方法实现场景代码实例
作者:Erneste 发布时间:2023-10-23 14:14:23
标签:Spring,Boot,异步,调用,场景
一、背景
项目中肯定会遇到异步调用其他方法的场景,比如有个计算过程,需要计算很多个指标的值,但是每个指标计算的效率快慢不同,如果采用同步执行的方式,运行这一个过程的时间是计算所有指标的时间之和。比如:
方法A:计算指标x,指标y,指标z的值,其中计算指标x需要1s,计算指标y需要2s,指标z需要3s。最终执行完方法A就是5s。
现在用异步的方式优化一下
方法A异步调用方法B,方法C,方法D,方法B,方法C,方法D分别计算指标x,指标y,指标z的值,那么最终执行完方法A的时间则是3s。
还有一种用途是当一个业务里面需要多个请求时,这时候异步并发请求所得到的回报远远是物有所值的。因为他是异步执行的,话不多说,一下是在springBoot里面使用并发请求;
二、spring boot中异步并发使用
2.1、appllication.yml
#****************集成Async线程池开始*******************
async: # Async线程池 配置
executor:
corepoolsize: 20
maxpoolsize: 25
queuecapacity: 40
keepaliveseconds: 200
threadnameprefix: appasync
awaitterminationseconds: 60
#*****************集成Async线程池结束******************
2.2、配置线程池
@Configuration
@EnableAsync
public class ExecutorConfig {
@Value("${async.executor.corepoolsize}")
private Integer corePoolSize;
@Value("${async.executor.maxpoolsize}")
private Integer maxPoolSize;
@Value("${async.executor.queuecapacity}")
private Integer queueCapacity;
@Value("${async.executor.keepaliveseconds}")
private Integer keepAliveSeconds;
@Value("${async.executor.threadnameprefix}")
private String threadNamePrefix;
@Value("${async.executor.awaitterminationseconds}")
private Integer awaitTerminationSeconds;
/**
* 线程池
*
* @return
*/
@Bean(name = "asyncExecutor")
public Executor asyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
// 基础线程数 corePoolSize: 10
executor.setCorePoolSize(corePoolSize);
// 最大线程数 maxPoolSize: 15
executor.setMaxPoolSize(maxPoolSize);
// 队列长度 queueCapacity: 25
executor.setQueueCapacity(queueCapacity);
// 线程池维护线程所允许的空闲时间,单位为秒 keepAliveSeconds: 200
executor.setKeepAliveSeconds(keepAliveSeconds);
// 线程名字 threadNamePrefix: appasync
executor.setThreadNamePrefix(threadNamePrefix);
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
// 等待所有任务都完成再继续销毁其他的Bean
executor.setWaitForTasksToCompleteOnShutdown(true);
// 线程池中任务的等待时间,如果超过这个时候还没有销毁就强制销毁,以确保应用最后能够被关闭,而不是阻塞住
executor.setAwaitTerminationSeconds(awaitTerminationSeconds);
executor.initialize();
return executor;
}
}
2.3、线程池监控(这个可有可无,主要是为了对线程池参数及时的调优)
@RestController
@Slf4j
@RequestMapping("/pubapi/asyncExecutor")
public class AsyncExecutorController extends BaseController {
@Resource(name = "asyncExecutor")
private Executor asyncExecutor;
@PostMapping("/monitor")public ResultBean<Map<String, Object>> getAsyncExecutorData() {
ResultBean<Map<String, Object>> resultBean = ResultBeanUtil.error500();
if (asyncExecutor == null) {
return resultBean;
}
try {
ThreadPoolTaskExecutor executorTask = (ThreadPoolTaskExecutor) asyncExecutor;
ThreadPoolExecutor executor = executorTask.getThreadPoolExecutor();
// 当前排队线程数
int queueSize = executor.getQueue().size();
// 当前活动线程数
int activeCount = executor.getActiveCount();
// 执行完线程数
long completedThreadCount = executor.getCompletedTaskCount();
// 总线程数
long taskCount = executor.getTaskCount();
// 初始线程数
int poolSize = executor.getPoolSize();
// 核心线程数
int corePoolSize = executor.getCorePoolSize();
// 线程池是否终止
boolean isTerminated = executor.isTerminated();
// 线城池是否关闭
boolean isShutdown = executor.isShutdown();
// 线程空闲时间
long keepAliveTime = executor.getKeepAliveTime(TimeUnit.MILLISECONDS);
// 最大允许线程数
long maximumPoolSize = executor.getMaximumPoolSize();
// 线程池中存在的最大线程数
long largestPoolSize = executor.getLargestPoolSize();
Map<String, Object> threadPoolData = new HashMap<>(18);
threadPoolData.put("当前排队线程数", queueSize);
threadPoolData.put("当前活动线程数", activeCount);
threadPoolData.put("执行完线程数", completedThreadCount);
threadPoolData.put("总线程数", taskCount);
threadPoolData.put("初始线程数", poolSize);
threadPoolData.put("核心线程数", corePoolSize);
threadPoolData.put("线程池是否终止", isTerminated);
threadPoolData.put("线城池是否关闭", isShutdown);
threadPoolData.put("线程空闲时间", keepAliveTime);
threadPoolData.put("最大允许线程数", maximumPoolSize);
threadPoolData.put("线程池中存在的最大线程数", largestPoolSize);
InetAddress inetAddress = IdWorker.getLocalHostLANAddress();
Map<String, Object> resultData = new HashMap<>(4);
resultData.put("ip", inetAddress.getHostAddress());
resultData.put("threadPoolData", threadPoolData);
resultBean = ResultBeanUtil.success("请求成功!", resultData);
} catch (Exception e) {
e.printStackTrace();
}
return resultBean;
}
}
2.4、代码中使用
public void getMap(){
/**
* 先将耗时的、相互之间无依赖的操作先执行,由于其执行结果暂时不是特别关注,所以
*/
Future<String> futureA = functionA();
Future<String> futureB = functionB();
/**
* 执行其他的操作,其实functionA(),functionB()也在工作
*/
aaa();
/**
* 获取异步的结果,然后计算
*/
try {
String resultA =futureA.get();
String resuleB = futureB.get();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (ExecutionException e) {
e.printStackTrace();
}
}
public Future<String> functionA (){
Future<String> future = null;
try {
Thread.sleep(5000);
future = new AsyncResult<String>("functionA");
} catch (InterruptedException e) {
e.printStackTrace();
}
return future;
}
public Future<String> functionB (){
Future<String> future = null;
try {
Thread.sleep(3000);
future = new AsyncResult<String>("functionB");
} catch (InterruptedException e) {
e.printStackTrace();
}
return future;
}
public void aaa(){
System.out.println("我是");
}
来源:https://www.cnblogs.com/DDgougou/p/11964425.html


猜你喜欢
- 首先我们先看下效果图实现思路这是两张前后对比图,右边第二张图里面的已抢光标签图片当已经没有商品的时候就会显示了,在每个图片的中心位置,第一想
- 因为在Action的execute方法声明时就抛出了Exception异常,所以我们无需再execute方法中捕捉异常,仅需在struts.
- 分享一个小技巧:在日常开发中有时候需要切换到另外的一个分支,但在某些条件下当前的分支上存在一些文件尚未提交,这时候就需要使用到idea自带的
- 在android移动端的开发中,首页轮播图是一个特别常见的功能,所以今天就来将最近写的一个小demo记录一下。首先当然是新建一个项目代码如下
- 一、模拟业务需求假设我们现在需要在我们的系统中导入一批关于学生信息的Excel的数据,其主要的信息有:学号、姓名、年龄、性别等等,在导入系统
- 引言大家应该都知道,对Excel表格设置分页对我们预览、打印文档时是很方便的,特别是一些包含很多复杂数据的、不规则的表格,为保证打印时每一页
- 如果需要实现跨服务器上传文件,就是将我们本地的文件上传到资源服务器上,比较好的办法就是通过ftp上传。这里是结合SpringMVC+ftp的
- java调用外部程序的方法 在一个java应用中,可能会遇到这样的需求,就是需要调用一些外部的应用做一些处理,比如调用excel,
- 1.底层网络接口采用apache的httpclient连接池框架; 2.图片缓存采用基于LRU的算法; 3.网络接口采用监听者模式; 4.包
- 前言最近使用QT中的QTextEdit控件,作为实时数据显示的UI,在一次写入超过多少k的时候循环写入则会卡顿,网上也没有什么好的解决方案,
- 1.1 概述分布式系统:分布式系统指由很多台计算机组成的一个整体!这个整体一致对外,并且处理同一请求!系统对内透明,对外不透明!内部的每台计
- Glide开源框架是Google推荐的图片加载和缓框架,其在Github上的开源地址是:https://github.com/bumptec
- 大家可能在做app的时候,或多或少需要使用联系人,而根据google提供的api,你需要编写大量的代码,例如首先需要查询数据库,涉及到数据库
- 本文实例为大家分享了C#遍历文件夹获取指定后缀名文件的具体代码,供大家参考,具体内容如下问题描述:项目需要,要进行某文件夹下所有shp数据的
- 一、使用策略枚举来优化if-else看到网上蛮多人推荐使用策略模式来优化if-else,但我总觉得,搞一堆策略类来优化大批量if-else,
- 本文实例为大家分享了C语言实现航空订票系统的具体代码,供大家参考,具体内容如下大一写的时候没有写注释,后来也懒得加了。在这里说一下读写文件的
- 现在Web开发越来越倾向于前后端分离,前端使用AngularJS,React,Vue等,部署在NodeJS上,后面采用SpringBoot发
- 在之前的C#版本中, 如果我们想要进行异步的Udp, 需要单开线程接收消息, C#7.1开始, 我们可以使用async/await关键字来编
- 前言但是没有合理的架构,大家写出来的代码很可能是一大堆的复制粘贴。比如十几个页面,都有这个关注按钮。然后,你是不是也要写十几个地方呢 然后修
- 前言在逛网易新闻时,发现列表中的广告在你滑动的时候会有一个3D旋转的交互引你的注意,不得不说这些产品为了让用户看广告花样百出,那么今天我们就