国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發 > Java > 正文

Spring Boot集成教程之異步調用Async

2024-07-14 08:40:08
字體:
來源:轉載
供稿:網友

前言

本文主要給大家介紹了關于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

異步調用成功,并且在所有任務都完成時程序才返回了結果!

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對VeVb武林網的支持。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 宝兴县| 津南区| 喀什市| 闽侯县| 庄浪县| 台安县| 井陉县| 甘肃省| 西平县| 柯坪县| 舒兰市| 乐都县| 阳原县| 临邑县| 桐梓县| 潞西市| 东源县| 阿鲁科尔沁旗| 灌云县| 顺昌县| 阿鲁科尔沁旗| 油尖旺区| 衡东县| 新建县| 游戏| 鲁山县| 乌鲁木齐县| 丰县| 湘乡市| 手机| 德庆县| 余庆县| 兴安县| 洞头县| 合肥市| 渑池县| 逊克县| 太康县| 南江县| 双流县| 体育|