Spring Boot集成教程之异步调用Async

前言

本文主要给大家介绍了关于Spring Boot集成之异步调用Async的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧。

什么是异步调用?

异步调用是相对于同步调用而言的,同步调用是指程序按预定顺序一步步执行,每一步必须等到上一步执行完后才能执行,异步调用则无需等待上一步程序执行完即可执行。

异步处理方式

  • 调用之后,不返回任何数据。
  • 调用之后,返回数据,通过Future来获取返回数据

如何实现异步调用?

多线程,这是很多人第一眼想到的关键词,没错,多线程就是一种实现异步调用的方式。

在非spring目项目中我们要实现异步调用的就是使用多线程方式,可以自己实现Runable接口或者集成Thread类,或者使用jdk1.5以上提供了的Executors线程池。

StrngBoot中则提供了很方便的方式执行异步调用。

按照官方示例开撸

代码入下

maven依赖:

<parent>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-parent</artifactId>
 <version>1.5.3.RELEASE</version>
</parent>
<dependencies>
 <dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-web</artifactId>
 </dependency>
</dependencies> 

启动类:添加@EnableAsync注解

@SpringBootApplication
@EnableAsync
public class Application{ 

 public static void main(String[] args) {
  SpringApplication.run(Application.class, args);
 }
} 

Controller

只需在需要异步执行方法上添加@Async注解

@RestController
@RequestMapping("")
public class AsyncTaskController {
 @RequestMapping("")
 public String doTask() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  this.task1();
  this.task2();
  this.task3();
  long currentTimeMillis1 = System.currentTimeMillis();
  return "task任务总耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms";
 } 

 @Async
 public void task1() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(1000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task1任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
 } 

 @Async
 public void task2() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(2000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task2任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
 }
 @Async
 public void task3() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(3000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task3任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
 }
} 

main函数运行spirngboot项目,启动完成后浏览器访问:http://localhost:8080/

控制台:

task1任务耗时:1012ms
task2任务耗时:2009ms
task3任务耗时:3004ms 

等了一段浏览器时候输出入下:

task任务总耗时:6002ms 

异步并没有执行!

难道是代码写错了?反复检查了好几遍,并没有发现什么明显错误,想起spring对@Transactional注解时也有类似问题,spring扫描时具有@Transactional注解方法的类时,是生成一个代理类,由代理类去开启关闭事务,而在同一个类中,方法调用是在类体内执行的,spring无法截获这个方法调用。

豁然开朗,将异步任务单独放到一个类中,调整代码入下:

Controller

@RequestMapping("")
@RestController
public class AsyncTaskController {
 @Autowired
 private AsyncTask asyncTask;
 @RequestMapping("")
 public String doTask() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  asyncTask.task1();
  asyncTask.task2();
  asyncTask.task3();
  long currentTimeMillis1 = System.currentTimeMillis();
  return "task任务总耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms"; 

 }
} 

异步任务类

@Component
public class AsyncTask {
 @Async
 public void task1() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(1000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task1任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
 } 

 @Async
 public void task2() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(2000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task2任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
 }
 @Async
 public void task3() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(3000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task3任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
 }
} 

控制台:

task1任务耗时:1012ms
task2任务耗时:2009ms
task3任务耗时:3004ms 

访问浏览器结果入下:

task任务总耗时:19ms 

异步调用成功!

如何知道三个异步任务什么时候执行完,执行的结果怎样呢?可以采用添加Fature回调方式判断

代码入下:

异步任务类

@Component
public class AsyncTask {
 @Async
 public Future<String> task1() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(1000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task1任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
  return new AsyncResult<String>("task1执行完毕");
 } 

 @Async
 public Future<String> task2() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(2000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task2任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
  return new AsyncResult<String>("task2执行完毕");
 }
 @Async
 public Future<String> task3() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Thread.sleep(3000);
  long currentTimeMillis1 = System.currentTimeMillis();
  System.out.println("task3任务耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms");
  return new AsyncResult<String>("task3执行完毕");
 }
} 

Controller

@RequestMapping("")
@RestController
public class AsyncTaskController {
 @Autowired
 private AsyncTask asyncTask;
 @RequestMapping("")
 public String doTask() throws InterruptedException{
  long currentTimeMillis = System.currentTimeMillis();
  Future<String> task1 = asyncTask.task1();
  Future<String> task2 = asyncTask.task2();
  Future<String> task3 = asyncTask.task3();
  String result = null;
  for (;;) {
   if(task1.isDone() && task2.isDone() && task3.isDone()) {
    // 三个任务都调用完成,退出循环等待
    break;
   }
   Thread.sleep(1000);
  }
  long currentTimeMillis1 = System.currentTimeMillis();
  result = "task任务总耗时:"+(currentTimeMillis1-currentTimeMillis)+"ms";
  return result;
 }
} 

控制台输出:

task1任务耗时:1000ms
task2任务耗时:2001ms
task3任务耗时:3001ms 

浏览器输出:

task任务总耗时:4015ms

异步调用成功,并且在所有任务都完成时程序才返回了结果!

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

您可能感兴趣的文章:

  • 深入理解spring boot异步调用方式@Async
  • spring boot 使用@Async实现异步调用方法
  • spring boot中使用@Async实现异步调用任务
  • spring boot使用自定义配置的线程池执行Async异步任务
  • Spring Boot 使用WebAsyncTask异步返回结果
  • spring boot使用自定义的线程池执行Async任务
  • spring boot异步(Async)任务调度实现方法
(0)

相关推荐

  • Spring Boot 使用WebAsyncTask异步返回结果

    在Spring Boot中(Spring MVC)下请求默认都是同步的,一个请求过去到结束都是由一个线程负责的,很多时候为了能够提高吞吐量,需要将一些操作异步化,除了一些耗时的业务逻辑可以异步化,我们的查询接口也是可以做到异步执行. 一个请求到服务上,是用的web容器的线程接收的,比如线程http-nio-8084-exec-1 我们可以使用WebAsyncTask将这个请求分发给一个新的线程去执行,http-nio-8084-exec-1可以去接收其他请求的处理.一旦WebAsyncTask返

  • spring boot异步(Async)任务调度实现方法

    在没有使用spring boot之前,我们的做法是在配置文件中定义一个任务池,然后将@Async注解的任务丢到任务池中去执行,那么在spring boot中,怎么来实现异步任务的调用了,方法更简单. 我们还是结合前面 spring boot整合JMS(ActiveMQ实现) 这篇博客里面的代码来实现. 一.功能说明 消费者在监听到队列里面的消息时,将接收消息的任务作为异步任务处理. 二.代码修改 消费者1: package com.chhliu.springboot.jms; import or

  • spring boot使用自定义的线程池执行Async任务

    在前面的博客中,//www.jb51.net/article/134866.htm 我们使用了spring boot的异步操作,当时,我们使用的是默认的线程池,但是,如果我们想根据项目来定制自己的线程池了,下面就来说说,如何定制线程池! 一.增加配置属性类 package com.chhliu.springboot.async.configuration; import org.springframework.boot.context.properties.ConfigurationProper

  • spring boot使用自定义配置的线程池执行Async异步任务

    在前面的博客中,http://www.jb51.net/article/106718.htm 我们使用了spring boot的异步操作,当时,我们使用的是默认的线程池,但是,如果我们想根据项目来定制自己的线程池了,下面就来说说,如何定制线程池! 一.增加配置属性类 package com.chhliu.springboot.async.configuration; import org.springframework.boot.context.properties.ConfigurationP

  • spring boot 使用@Async实现异步调用方法

    使用@Async实现异步调用 什么是"异步调用"与"同步调用" "同步调用"就是程序按照一定的顺序依次执行,,每一行程序代码必须等上一行代码执行完毕才能执行:"异步调用"则是只要上一行代码执行,无需等待结果的返回就开始执行本身任务. 通常情况下,"同步调用"执行程序所花费的时间比较多,执行效率比较差.所以,在代码本身不存在依赖关系的话,我们可以考虑通过"异步调用"的方式来并发执行. &q

  • spring boot中使用@Async实现异步调用任务

    什么是"异步调用"? "异步调用"对应的是"同步调用",同步调用指程序按照定义顺序依次执行,每一行程序都必须等待上一行程序执行完成之后才能执行:异步调用指程序在顺序执行时,不等待异步调用的语句返回结果就执行后面的程序.  同步调用 下面通过一个简单示例来直观的理解什么是同步调用: 定义Task类,创建三个处理函数分别模拟三个执行任务的操作,操作消耗时间随机取(10秒内) package com.kfit.task; import java.uti

  • 深入理解spring boot异步调用方式@Async

    本文主要给大家介绍了关于spring boot异步调用方式@Async的相关内容,分享出来供大家参考学习,下面来一起看看详细的介绍: 1.使用背景 在日常开发的项目中,当访问其他人的接口较慢或者做耗时任务时,不想程序一直卡在耗时任务上,想程序能够并行执行,我们可以使用多线程来并行的处理任务,也可以使用spring提供的异步处理方式@Async. 2.异步处理方式 调用之后,不返回任何数据. 调用之后,返回数据,通过Future来获取返回数据 3.@Async不返回数据 使用@EnableAsyn

  • Spring Boot集成教程之异步调用Async

    前言 本文主要给大家介绍了关于Spring Boot集成之异步调用Async的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 什么是异步调用? 异步调用是相对于同步调用而言的,同步调用是指程序按预定顺序一步步执行,每一步必须等到上一步执行完后才能执行,异步调用则无需等待上一步程序执行完即可执行. 异步处理方式 调用之后,不返回任何数据. 调用之后,返回数据,通过Future来获取返回数据 如何实现异步调用? 多线程,这是很多人第一眼想到的关键词,没错,多线程就是一种实

  • Spring Boot 集成Shiro的多realm实现以及shiro基本入门教程

    情景 我的项目中有六个用户角色(学校管理员,学生等),需要进行分别登陆.如果在一个realm中,对controller封装好的Token进行Service验证,需要在此realm中注入六个数据库操作对象,然后写一堆if语句来判断应该使用那个Service服务,然后再在验证方法(doGetAuthorizationInfo)中写一堆if来进行分别授权,这样写不仅会让代码可读性会非常低而且很难后期维护修改(刚写完的时候只有上帝和你能看懂你写的是什么,一个月之后你写的是什么就只有上帝能看懂了). 所以

  • Spring Boot 集成MyBatis 教程详解

    Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程.该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置.通过这种方式,Spring Boot致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者. 在集成MyBatis前,我们先配置一个druid数据源. Spring Boot 系列 1.Spring Boot 入门 2.Spring Boot 属性配置

  • spring boot集成rabbitmq的实例教程

    一.RabbitMQ的介绍 RabbitMQ是消息中间件的一种,消息中间件即分布式系统中完成消息的发送和接收的基础软件.这些软件有很多,包括ActiveMQ(apache公司的),RocketMQ(阿里巴巴公司的,现已经转让给apache). 消息中间件的工作过程可以用生产者消费者模型来表示.即,生产者不断的向消息队列发送信息,而消费者从消息队列中消费信息.具体过程如下: 从上图可看出,对于消息队列来说,生产者,消息队列,消费者是最重要的三个概念,生产者发消息到消息队列中去,消费者监听指定的消息

  • spring boot集成shiro详细教程(小结)

    我们开发时候有时候要把传统spring shiro转成spring boot项目,或者直接集成,name我们要搞清楚一个知识,就是 xml配置和spring bean代码配置的关系,这一点很重要,因为spring boot是没有xml配置文件的(也不绝对,spring boot也是可以引用xml配置的) 引入依赖: <dependency> <artifactId>ehcache-core</artifactId> <groupId>net.sf.ehcac

  • Spring boot集成RabbitMQ的示例代码

    RabbitMQ简介 RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统 MQ全称为Message Queue, 消息队列(MQ)是一种应用程序对应用程序的通信方法.应用程序通过读写出入队列的消息(针对应用程序的数据)来通信,而无需专用连接来链接它们.消息传递指的是程序之间通过在消息中发送数据进行通信,而不是通过直接调用彼此来通信,直接调用通常是用于诸如远程过程调用的技术.排队指的是应用程序通过 队列来通信.队列的使用除去了接收和发送应用程序同时执行的要求. AMQP就是一个协议

  • springboot实现异步调用@Async的示例

    在后端开发中经常遇到一些耗时或者第三方系统调用的情况,我们知道Java程序一般的执行流程是顺序执行(不考虑多线程并发的情况),但是顺序执行的效率肯定是无法达到我们的预期的,这时就期望可以并行执行,常规的做法是使用多线程或线程池,需要额外编写代码实现.在spring3.0后引入了@Async注解,使用该注解可以达到线程池的执行效果,而且在开发上非常简单. 一.概述 springboot是基于spring框架的,在springboot环境下演示@Async注解的使用方式.先看下该注解的定义, @Ta

  • 深入研究spring boot集成kafka之spring-kafka底层原理

    目录 前言 简单集成 引入依赖 添加配置 测试发送和接收 Spring-kafka-test嵌入式KafkaServer 引入依赖 启动服务 创建新的Topic 程序启动时创建TOPIC 代码逻辑中创建 PS:其他的方式创建TOPIC 引入依赖 api方式创建 命令方式创建 消息发送之KafkaTemplate探秘 获取发送结果 异步获取 同步获取 KAFKA事务消息 REPLYINGKAFKATEMPLATE获得消息回复 Spring-kafka消息消费用法探秘 @KAFKALISTENER的

  • Spring Boot集成JavaMailSender发送邮件功能的实现

    目录 前言 集成步骤 添加依赖 邮件配置信息 邮件配置类 代码实现 发送简单邮件 邮件实体类 业务实现类 测试类 扩展功能 发送Html内容的邮件 业务实现类 测试类 发送带附件邮件 业务实现类 测试类 发送模板邮件 添加依赖 模板内容 业务实现类 测试示例 总结 前言 项目中邮件发送为常用功能之一,例如注册成功后需要发送激活邮件,账号的会员到期提示,每日报表统计等功能都需要自动发送邮件,本文将讲解Spring Boot如何实现邮件发送功能. 集成步骤 添加依赖 <dependency> &l

  • Spring Boot集成MyBatis访问数据库的方法

    基于spring boot开发的微服务应用,与MyBatis如何集成? 集成方法 可行的方法有: 1.基于XML或者Java Config,构建必需的对象,配置MyBatis. 2.使用MyBatis官方提供的组件,实现MyBatis的集成. 方法一 建议参考如下文章,完成集成的验证. MyBatis学习 之 一.MyBatis简介与配置MyBatis+Spring+MySql 基于Spring + Spring MVC + Mybatis 高性能web构建 spring与mybatis三种整合

随机推荐