springboot中@EnableAsync與@Async註解使用java
@Async爲異步註解,放到方法上,表示調用該方法的線程與此方法異步執行,須要配合@EnableAsync註解使用。
spring
一、首先演示沒有@Async,即沒有異步執行的狀況springboot
- 建立一個普通類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的beanspa
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循環語句。從整個流程看屬於順序執行的。code
二、有@Async,即異步執行的狀況對象
- 建立一個普通類CountNumber,並注入到IOC容器中,並在該類的方法上標註@Async註解,表示該方法是異步執行的。blog
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註解,由於該註解中已經包含了上面兩個註解。get
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方法是與主線程是異步執行的。