轉自:https://www.cnblogs.com/UniqueColor/p/9295642.html
springboot中@EnableAsync與@Async注解使用
@Async為異步注解,放到方法上,表示調用該方法的線程與此方法異步執行,需要配合@EnableAsync注解使用。
1、首先演示沒有@Async,即沒有異步執行的情況
- 創建一個普通類CountNumber,並注入到IOC容器中
1 package com.example.demo;
2 import org.springframework.scheduling.annotation.Async;
3 import org.springframework.stereotype.Component;
4
5 @Component
6 public class CountNumber {
7
8 public void PrintNumber(){
9 for(int i=1; i<10; i++){
10 System.out.println("i = " + i);
11 }
12 }
13 }
- 在spring boot的啟動類中獲取IOC的bean
1 package com.example.demo;
2 import java.util.concurrent.TimeUnit;
3 import org.springframework.boot.SpringApplication;
4 import org.springframework.boot.autoconfigure.SpringBootApplication;
5 import org.springframework.context.ConfigurableApplicationContext;
6 import org.springframework.context.annotation.ComponentScan;
7
8 //@SpringBootApplication
9 @ComponentScan
10 public class Springboot3Application {
11
12 public static void main(String[] args) throws Exception {
13
14 ConfigurableApplicationContext context = SpringApplication.run(Springboot3Application.class, args);
15
16 context.getBean(CountNumber.class).PrintNumber();
17 for(int i=1; i<10; i++){
18 System.out.println("------------------");
19 }
20 context.close();
21 }
22 }
- 運行輸出結果:
i = 1 i = 2 i = 3 i = 4 i = 5 i = 6 i = 7 i = 8 i = 9 ------------------ ------------------ ------------------ ------------------ ------------------ ------------------ ------------------ ------------------ ------------------
從輸出結果中可以看出,啟動類先從IOC容器中獲取CountNumber的對象,然后執行該對象的PrintNumber方法,循環打印了9個數字,方法執行結束后,繼續回到啟動類中往下執行,因此開始執行for循環語句。從整個流程看屬於順序執行的。
2、有@Async,即異步執行的情況
- 創建一個普通類CountNumber,並注入到IOC容器中,並在該類的方法上標注@Async注解,表示該方法是異步執行的。
1 package com.example.demo;
2 import org.springframework.scheduling.annotation.Async;
3 import org.springframework.stereotype.Component;
4
5 @Component
6 public class CountNumber {
7 @Async
8 public void PrintNumber(){
9 for(int i=1; i<10; i++){
10 System.out.println("i = " + i);
11 }
12 }
13 }
- 從spring boot啟動類中獲取IOC中的bean,在啟動類上標注@EnableAsync注解,啟動@Async異步注解。或者在啟動類中只標注@SpringBootApplication注解,因為該注解中已經包含了上面兩個注解。
1 package com.example.demo;
2 import java.util.concurrent.TimeUnit;
3 import org.springframework.boot.SpringApplication;
4 import org.springframework.boot.autoconfigure.SpringBootApplication;
5 import org.springframework.context.ConfigurableApplicationContext;
6 import org.springframework.context.annotation.ComponentScan;
7 import org.springframework.scheduling.annotation.Async;
8 import org.springframework.scheduling.annotation.EnableAsync;
9
10 /*@SpringBootApplication注解與@ComponentScan、@EnableAsync注解達到相同的功效*/
11 //@SpringBootApplication
12 @ComponentScan
13 @EnableAsync
14 public class Springboot3Application {
15
16 public static void main(String[] args) throws Exception {
17
18 ConfigurableApplicationContext context = SpringApplication.run(Springboot3Application.class, args);
19
20 /*@Async和@EnableAsync配合使用*/
21 context.getBean(CountNumber.class).PrintNumber();
22 for(int i=1; i<10; i++){
23 TimeUnit.MICROSECONDS.sleep(1);
24 System.out.println("------------------");
25 }
26 context.close();
27 }
28 }
- 執行啟動類,輸出結果如下:
------------------ ------------------ ------------------ ------------------ ------------------ ------------------ i = 1 i = 2 i = 3 i = 4 i = 5 i = 6 i = 7 i = 8 i = 9 ------------------ ------------------ ------------------
從輸出結果中可以看出,spring boot在獲取到IOC中的CountNumber對象后,一方面繼續向下執行,執行for循環語句,另一方面獲取對象后,執行對象中的PrintNumber方法。因此PrintNumber方法是與主線程是異步執行的。

