springboot:使用异步注解@Async的那些坑

一、引言

在java后端开发中经常会碰到处理多个任务的情况,比如一个方法中要调用多个请求,然后把多个请求的结果合并后统一返回,一般情况下调用其他的请求一般都是同步的,也就是每个请求都是阻塞的,那么这个处理时间必定是很长的,有没有一种方法可以让多个请求异步处理那,答案是有的。

springboot中提供了很便利的方式可以解决上面的问题,那就是异步注解@Async。正确的使用该注解可以使你的程序飞起,相反如果使用不当那么并不会取到理想的效果。

二、获取异步执行结果

1、环境介绍

下面是我的controller,SyncController.java

  1. package com.atssg.controller;
  2. import com.atssg.service.MySyncService;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.springframework.beans.factory.annotation.Autowired;
  5. import org.springframework.web.bind.annotation.GetMapping;
  6. import org.springframework.web.bind.annotation.RequestMapping;
  7. import org.springframework.web.bind.annotation.RestController;
  8. @Slf4j
  9. @RestController
  10. @RequestMapping("/sync")
  11. public class SyncController {
  12. @Autowired
  13. private MySyncService syncService;
  14. @GetMapping(value = "/test")
  15. public String test() {
  16. String str=null;
  17. try {
  18. log.info("start");
  19. str = syncService.asyncMethod();
  20. log.info("str:{}", str);
  21. return str;
  22. } catch (Exception e) {
  23. e.printStackTrace();
  24. }
  25. return str;
  26. }
  27. }

在controller中就是调用下层的方法并返回,再看service层的类MySyncService.java

  1. package com.atssg.service;
  2. import org.springframework.beans.factory.annotation.Autowired;
  3. import org.springframework.stereotype.Service;
  4. import java.util.concurrent.ExecutionException;
  5. import java.util.concurrent.Future;
  6. @Service
  7. public class MySyncService {
  8. @Autowired
  9. private SyncService syncService;
  10. /**
  11. * 异步方法
  12. *
  13. * @return
  14. * @throws InterruptedException
  15. * @throws ExecutionException
  16. */
  17. public String asyncMethod() throws InterruptedException, ExecutionException {
  18. Future<String> result1 = syncService.method1("I");
  19. Future<String> result2 = syncService.method2("love");
  20. Future<String> result3 = syncService.method3("async");
  21. String str = result1.get();
  22. String str2 = result2.get();
  23. String str3 = result3.get();
  24. String result = str + str2 + str3;
  25. return result;
  26. }
  27. /**
  28. * 同步方法
  29. *
  30. * @return
  31. * @throws InterruptedException
  32. * @throws ExecutionException
  33. */
  34. public String syncMethod() throws InterruptedException, ExecutionException {
  35. /*同步写法*/
  36. String str = syncService.method1("I").get();
  37. String str2 = syncService.method2("love").get();
  38. String str3 = syncService.method3("async").get();
  39. return str + str2 + str3;
  40. }
  41. }

上面便是service类,仅仅是调用下次异步层的方法,并取得返回值。上面类中有两个方法,其写法也类似但结果却大不相同,后面详说。

下面是异步层的方法,SyncService.java

  1. package com.atssg.service;
  2. import org.springframework.scheduling.annotation.Async;
  3. import org.springframework.scheduling.annotation.AsyncResult;
  4. import org.springframework.stereotype.Service;
  5. import java.util.concurrent.Future;
  6. @Service
  7. @Async
  8. public class SyncService {
  9. //@Async
  10. public Future<String> method1(String str) throws InterruptedException {
  11. Thread.sleep(1000*10);
  12. return new AsyncResult<>( str);
  13. }
  14. //@Async
  15. public Future<String> method2(String str) throws InterruptedException {
  16. Thread.sleep(1000*5);
  17. return new AsyncResult<>(str);
  18. }
  19. // @Async
  20. public Future<String> method3(String str) throws InterruptedException {
  21. Thread.sleep(1000*15);
  22. return new AsyncResult<>(str);
  23. }
  24. }

该类使用@Async注解,表明该类中所有的方法都是异步执行的,其中@Async可修饰类也可以修饰方法。

这便是所有的环境。

2、错误的方式

在MySyncService中有两个方法,先看其中一个方法

  1. public String syncMethod() throws InterruptedException, ExecutionException {
  2. /*同步写法*/
  3. String str = syncService.method1("I").get();
  4. String str2 = syncService.method2("love").get();
  5. String str3 = syncService.method3("async").get();
  6. return str + str2 + str3;
  7. }

这种写法是调用异步方法后立即调用get()方法,即获取结果,下面看测试结果,在controllor中调用该方法,下面看执行结果

  1. 2021-08-21 11:06:28.612 INFO 3584 --- [nio-8080-exec-1] com.atssg.controller.SyncController : start
  2. 2021-08-21 11:06:58.651 INFO 3584 --- [nio-8080-exec-1] com.atssg.controller.SyncController : str:Iloveasync

可以看到共执行了30s,在异步层的方法中的三个方法如下,

  1. //@Async
  2. public Future<String> method1(String str) throws InterruptedException {
  3. Thread.sleep(1000*10);
  4. return new AsyncResult<>( str);
  5. }
  6. //@Async
  7. public Future<String> method2(String str) throws InterruptedException {
  8. Thread.sleep(1000*5);
  9. return new AsyncResult<>(str);
  10. }
  11. // @Async
  12. public Future<String> method3(String str) throws InterruptedException {
  13. Thread.sleep(1000*15);
  14. return new AsyncResult<>(str);
  15. }

可以看到这三个方法分别是睡眠10s、5s、15s,这就很好理解了syncMethod()方法中的写法是同步的,未达到异步的目的,切记调用完异步方法进接着调用get()方法不是异步的方式,而是同步的。

3、正确方式

上面看了错误的用法,下面看正确的方式,

  1. public String asyncMethod() throws InterruptedException, ExecutionException {
  2. Future<String> result1 = syncService.method1("I");
  3. Future<String> result2 = syncService.method2("love");
  4. Future<String> result3 = syncService.method3("async");
  5. String str = result1.get();
  6. String str2 = result2.get();
  7. String str3 = result3.get();
  8. String result = str + str2 + str3;
  9. return result;
  10. }

这种方式是首先调用异步方法,然后分别调用get()方法,取得执行结果。下面看测试结果

  1. 2021-08-21 11:17:23.516 INFO 3248 --- [nio-8080-exec-1] com.atssg.controller.SyncController : start
  2. 2021-08-21 11:17:38.535 INFO 3248 --- [nio-8080-exec-1] com.atssg.controller.SyncController : str:Iloveasync

执行时间未15s,这就很好解释了,异步层的三个方法,分别睡眠的时间是10s、5s、15s,既然是异步执行的,那么总的执行时间肯定是三个方法中最长的那个,符合测试结果。这才@Async正确的打开姿势。

三、异步执行@Async注解

@Async注解的定义如下,

  1. @Target({ElementType.TYPE, ElementType.METHOD})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Documented
  4. public @interface Async {
  5. String value() default "";
  6. }

可以看到该注解可以用在类及方法上,用在类上表示类中的所有方法都是异步的,用在方法上表示该方法是异步的。

四、总结

今天的文章分享到这里,主要分享了关于@Async注解在获取执行结果的时候的坑,一定要先调用异步方法,然后再调用get()方法,获取结果,其中get方法还有一个重载的,可以设置超时时间,即超过设置的超时时间便返回,不再等待,各位小伙伴可以自己试验。

  1. V get(long timeout, TimeUnit unit)
  2. throws InterruptedException, ExecutionException, TimeoutException;
  3. }

下次继续分享有关@Async注解使用的一些小细节,欢迎持续关注。

推荐阅读

[springboot:异步调用@Async]:

springboot:使用异步注解@Async的那些坑的更多相关文章

  1. springboot:嵌套使用异步注解@Async还会异步执行吗

    一.引言 在前边的文章<[springboot:使用异步注解@Async的那些坑>中介绍了使用@Async注解获取任务执行结果的错误用法,今天来分享下另外一种常见的错误. 二.代码演示 下 ...

  2. springboot:使用异步注解@Async的前世今生

    在前边的文章中,和小伙伴一起认识了异步执行的好处,以及如何进行异步开发,对,就是使用@Async注解,在使用异步注解@Async的过程中也存在一些坑,不过通过正确的打开方式也可以很好的避免,今天想和大 ...

  3. Spring中异步注解@Async的使用、原理及使用时可能导致的问题

    前言 其实最近都在研究事务相关的内容,之所以写这么一篇文章是因为前面写了一篇关于循环依赖的文章: <面试必杀技,讲一讲Spring中的循环依赖> 然后,很多同学碰到了下面这个问题,添加了S ...

  4. springboot之异步调用@Async

    原文:http://www.cnblogs.com/xuwenjin/p/8858050.html 引言: 在Java应用中,绝大多数情况下都是通过同步的方式来实现交互处理的:但是在处理与第三方系统交 ...

  5. springboot:异步调用@Async

    在后端开发中经常遇到一些耗时或者第三方系统调用的情况,我们知道Java程序一般的执行流程是顺序执行(不考虑多线程并发的情况),但是顺序执行的效率肯定是无法达到我们的预期的,这时就期望可以并行执行,常规 ...

  6. 关于Dubbo和Spring异步注解@Async的冲突

    项目中难免会有异步处理的需求,像异步记录日志啦,异步发送邮件啦,而Dubbo又是现在主流的分布式框架,所有异步+Dubbo的组合是再所难免的 但博主是实践中发现Dubbo的服务并不能很好的跟Sprin ...

  7. Java中异步注解@Async的陷阱

    或许,你在Java后端添加异步过程时会这样处理,然后摇摇大摆.灰溜溜地闪,而实际的运行结果却并不是我们期望的那样.那么,现在就将试验结果记录如下,以便少走弯路. (一)在Controller层的公开接 ...

  8. SpringBoot使用异步线程池实现生产环境批量数据推送

    前言 SpringBoot使用异步线程池: 1.编写线程池配置类,自定义一个线程池: 2.定义一个异步服务: 3.使用@Async注解指向定义的线程池: 这里以我工作中使用过的一个案例来做描述,我所在 ...

  9. @Async异步注解与SpringBoot结合使用

    当你在service层需要启动异步线程去执行某些分支任务,又不希望显式使用Thread等线程相关类,只想专注于实现业务逻辑代码开发,可以使用@Async异步注解. 1. 使用@Async 异步注解 C ...

随机推荐

  1. 1.3.7、通过QueryParam匹配

    server: port: 8080 spring: application: name: gateway cloud: gateway: routes: - id: guo-system4 uri: ...

  2. 玩Aarch64最方便的方法

    译至:http://d.hatena.ne.jp/embedded/20140819/p1 虽然Aarch64(ARM64)的板子还很难到手.但通过使用qemu就能执行Aarch64的用户空间程序.利 ...

  3. 什么是 Acunetix 目标知识库

    随着Acunetix 的最新更新,我们引入了一个称为目标知识库的新功能.每次扫描目标时,Acunetix 都会收集并存储有关它的信息.此信息包括构成站点结构的路径.表单的位置及其输入.Web 应用程序 ...

  4. 如何统计自动化测试用例的ROI 【投入产出比/投资回报率】

    一. 自动化测试的投入有哪些? 1. 自动化测试的软件平台投入 自动化测试平台的开发时间,实施时间. 2.  自动化测试的框架开发投入+框架维护的投入 开发自动化测试脚本使用的框架,例如通过一些现有框 ...

  5. android体温登记APP开发过程+问题汇总+源码

    源码上传至https://github.com/durtime/myproject下的temperature 实际效果:   开发过程 1.首先进行布局文件的编写,布局前台页面 2.布置两个按钮,一个 ...

  6. [刘阳Java]_CSS普通菜单制作

    简单给大家介绍一下CSS普通菜单制作,先看图 功能很简单 UL制作菜单 鼠标进入LI的CSS伪类,实现菜单弹出效果 源码如下 <!DOCTYPE html> <html> &l ...

  7. 【算法学习笔记】概率与期望DP

    本文学习自 Sengxian 学长的博客 之前也在CF上写了一些概率DP的题并做过总结 建议阅读完本文再去接着阅读这篇文章:Here 前言 单纯只用到概率的题并不是很多,从现有的 OI/ACM 比赛中 ...

  8. CSAPP:datalab实验记录

    CSAPP:datalab实验记录 bitXor /* * bitXor - x^y using only ~ and & * Example: bitXor(4, 5) = 1 * Lega ...

  9. PAT乙级:1083 是否存在相等的差 (20分)

    PAT乙级:1083 是否存在相等的差 (20分) 题干 给定 N 张卡片,正面分别写上 1.2.--.N,然后全部翻面,洗牌,在背面分别写上 1.2.--.N.将每张牌的正反两面数字相减(大减小), ...

  10. 小师妹学IO系列文章集合-附PDF下载

    目录 第一章 IO的本质 IO的本质 DMA和虚拟地址空间 IO的分类 IO和NIO的区别 总结 第二章 try with和它的底层原理 简介 IO关闭的问题 使用try with resource ...