什么是异步调用

“异步调用”对应的是“同步调用”,同步调用指程序按照定义顺序依次执行,每一行程序都必须等待上一行程序执行完成之后才能执行;异步调用指程序在顺序执行时,不等待异步调用的语句返回结果就执行后面的程序。

同步调用

下面通过一个简单示例来直观的理解什么是同步调用:

定义Task类,创建三个处理函数分别模拟三个执行任务的操作,操作消耗时间随机取(10秒内)

package com.kfit.task;

import java.util.Random;

import org.springframework.stereotype.Component;

/**

* 定义3个任务

* @author Angel(QQ:412887952)

* @version v.0.1

*/

@Component

publicclass Task1 {

//定义一个随机对象.

publicstatic Random random =new Random();

//任务一;

publicvoid doTaskOne() throws Exception {

System.out.println("开始做任务一");

longstart = System.currentTimeMillis();

Thread.sleep(random.nextInt(10000));

longend = System.currentTimeMillis();

System.out.println("完成任务一,耗时:" + (end - start) + "毫秒");

}

//任务二;

publicvoid doTaskTwo() throws Exception {

System.out.println("开始做任务二");

longstart = System.currentTimeMillis();

Thread.sleep(random.nextInt(10000));

longend = System.currentTimeMillis();

System.out.println("完成任务二,耗时:" + (end - start) + "毫秒");

}

//任务3;

publicvoid doTaskThree() throws Exception {

System.out.println("开始做任务三");

longstart = System.currentTimeMillis();

Thread.sleep(random.nextInt(10000));

longend = System.currentTimeMillis();

System.out.println("完成任务三,耗时:" + (end - start) + "毫秒");

}

}

编写一个访问方法:

//测试task1.

@RequestMapping("/task1")

public String task1() throws Exception{

task1.doTaskOne();

task1.doTaskTwo();

task1.doTaskThree();

return"task1";

}

运行可以看到类似如下输出:

开始做任务一

完成任务一,耗时:4156毫秒

开始做任务二

完成任务二,耗时:557毫秒

开始做任务三

完成任务三,耗时:6171毫秒

异步调用

上述的同步调用虽然顺利的执行完了三个任务,但是可以看到执行时间比较长,若这三个任务本身之间不存在依赖关系,可以并发执行的话,同步调用在执行效率方面就比较差,可以考虑通过异步调用的方式来并发执行。

在Spring Boot中,我们只需要通过使用@Async注解就能简单的将原来的同步函数变为异步函数,Task类改在为如下模式:

package com.kfit.task;

import java.util.Random;

import org.springframework.scheduling.annotation.Async;

import org.springframework.stereotype.Component;

/**

* 定义3个任务

* @author Angel(QQ:412887952)

* @version v.0.1

*/

@Component

publicclass Task2 {

//定义一个随机对象.

publicstatic Random random =new Random();

//任务一;

@Async

publicvoid doTaskOne() throws Exception {

System.out.println("开始做任务一");

longstart = System.currentTimeMillis();

Thread.sleep(random.nextInt(10000));

longend = System.currentTimeMillis();

System.out.println("完成任务一,耗时:" + (end - start) + "毫秒");

}

//任务二;

@Async

publicvoid doTaskTwo() throws Exception {

System.out.println("开始做任务二");

longstart = System.currentTimeMillis();

Thread.sleep(random.nextInt(10000));

longend = System.currentTimeMillis();

System.out.println("完成任务二,耗时:" + (end - start) + "毫秒");

}

//任务3;

@Async

publicvoid doTaskThree() throws Exception {

System.out.println("开始做任务三");

longstart = System.currentTimeMillis();

Thread.sleep(random.nextInt(10000));

longend = System.currentTimeMillis();

System.out.println("完成任务三,耗时:" + (end - start) + "毫秒");

}

}

为了让@Async注解能够生效,还需要在Spring Boot的主程序中配置@EnableAsync,如下所示:

@SpringBootApplication

@EnableAsync

publicclass App {

//省略其它代码…

}

编写测试方法:

//测试task2.

@RequestMapping("/task2")

public String task2() throws Exception{

task2.doTaskOne();

task2.doTaskTwo();

task2.doTaskThree();

return"task2";

}

此时可以反复执行单元测试,您可能会遇到各种不同的结果,比如:

开始做任务一

开始做任务二

开始做任务三

完成任务三,耗时:57毫秒

完成任务二,耗时:3621毫秒

完成任务一,耗时:7419毫秒

Spring Boot 系列博客】

54. spring boot日志升级篇—logback【从零开始学Spring Boot】

52. spring boot日志升级篇—log4j多环境不同日志级别的控制【从零开始学Spring Boot】

51. spring boot属性文件之多环境配置【从零开始学Spring Boot】

50. Spring Boot日志升级篇—log4j【从零开始学Spring Boot】

49. spring boot日志升级篇—理论【从零开始学Spring Boot】

48. spring boot单元测试restfull API【从零开始学Spring Boot】

47. Spring Boot发送邮件【从零开始学Spring Boot】

46. Spring Boot中使用AOP统一处理Web请求日志

45. Spring Boot MyBatis连接Mysql数据库【从零开始学Spring Boot】

44. Spring Boot日志记录SLF4J【从零开始学Spring Boot】

43. Spring Boot动态数据源(多数据源自动切换)【从零开始学Spring Boot】

42. Spring Boot多数据源【从零开始学Spring Boot】

41. Spring Boot 使用Java代码创建Bean并注册到Spring中【从零开始学Spring Boot】

40. springboot + devtools(热部署)【从零开始学Spring Boot】

39.4 Spring Boot Shiro权限管理【从零开始学Spring Boot】

39.3 Spring Boot Shiro权限管理【从零开始学Spring Boot】

39.2. Spring Boot Shiro权限管理【从零开始学Spring Boot】

39.1 Spring Boot Shiro权限管理【从零开始学Spring Boot】

38 Spring Boot分布式Session状态保存Redis【从零开始学Spring Boot】

37 Spring Boot集成EHCache实现缓存机制【从零开始学Spring Boot】

36 Spring Boot Cache理论篇【从零开始学Spring Boot】

35 Spring Boot集成Redis实现缓存机制【从零开始学Spring Boot】

34Spring Boot的启动器Starter详解【从零开始学Spring Boot】

33 Spring Boot 监控和管理生产环境【从零开始学Spring Boot】

32 Spring Boot使用@SpringBootApplication注解【从零开始学Spring Boot】

31 Spring Boot导入XML配置【从零开始学Spring Boot】

更多查看博客: http://412887952-qq-com.iteye.com/

最新文章

  1. 【Beta】Daily Scrum Meeting第一次
  2. ASP.NET MVC Ajax.ActionLink 简单用法
  3. dojo/dom-class源码学习
  4. Moon.Orm 5.0(MQL版)分页功能的设计(求指教,邀请您的加入)
  5. openjudge8469特殊密码锁[贪心]
  6. vs2013使用C#6.0
  7. 用Feature的方式删除SharePoint2010的Page中重复的WebPart
  8. winform 计算器
  9. Struts2 Spring Hibernate等各个版本下载推荐
  10. poj 1265 Area (Pick定理+求面积)
  11. myeclipse 清理项目缓存的几大步骤
  12. js的捕捉事件,冒泡事件
  13. leanchat-android
  14. HDFS(Hadoop Distributed File System )
  15. stream转byte数组几种方式
  16. 超级密码(dfs)
  17. Linux查看用于终止进程命令
  18. css子元素添加绝对定位,不添加top、left会有影响吗???
  19. erlang并发编程(二)
  20. 前端_标签01_input标签

热门文章

  1. BZOJ 1396||2865 识别子串
  2. DP(DAG) UVA 437 The Tower of Babylon
  3. 题解报告:hdu 1229 还是A+B
  4. 442 Find All Duplicates in an Array 数组中重复的数据
  5. AJPFX关于IO流的简单总结
  6. VS2012出现加载失败时的解决办法 win7同样适用
  7. Android习惯——给全部Activity添加集合管理
  8. iOS-UI控件之UITableView(二)- 自定义不等高的cell
  9. PostgreSQL执行机制的初步学习
  10. Katalon Studio 安装 配置 简单使用