您好,登錄后才能下訂單哦!
這篇文章給大家介紹springboot中的@Async怎么實現異步調用,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
一、概述
springboot是基于spring框架的,在springboot環境下演示@Async注解的使用方式。先看下該注解的定義,
@Target({ElementType.METHOD, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface Async { /** * A qualifier value for the specified asynchronous operation(s). * <p>May be used to determine the target executor to be used when executing this * method, matching the qualifier value (or the bean name) of a specific * {@link java.util.concurrent.Executor Executor} or * {@link org.springframework.core.task.TaskExecutor TaskExecutor} * bean definition. * <p>When specified on a class level {@code @Async} annotation, indicates that the * given executor should be used for all methods within the class. Method level use * of {@code Async#value} always overrides any value set at the class level. * @since 3.1.2 */ String value() default ""; }
可以看到該注解只有一個屬性,那就是value,從注釋上知道value指定的是執行該任務的線程池,也就是說我們可以使用子定義的線程池執行我們的任務,而不是系統默認的。在看該注解上的注解,
@Target({ElementType.METHOD, ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented
也就是說該注解可以用在方法和類上。標記在類上表示類中的所有方法都以異步方式執行,也就是提交到線程池執行。
上面簡單對@Async注解進行了解釋,下面看用法。
在springboot中要使用@Async注解必須在springboot啟動類上使用@EnableAsync注解,開啟@Async注解的自動配置,如下,
package com.example.demo; import com.example.demo.properties.ApplicationPro; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.scheduling.annotation.EnableAsync; import org.springframework.scheduling.annotation.EnableScheduling; @SpringBootApplication @EnableConfigurationProperties({ApplicationPro.class}) //開啟@Async注解的自動配置 @EnableAsync public class DemoApplication { public static void main(String[] args) { SpringApplication.run(DemoApplication.class, args); } }
只有在啟動類上使用@EnableAsync注解,@Async注解才會生效。
上面使用@EnableAsync注解已經開啟了對@Async注解的配置,下面看具體的異步調用類,
package com.example.demo.service; import com.example.demo.Student; import org.springframework.scheduling.annotation.Async; import org.springframework.scheduling.annotation.AsyncResult; import org.springframework.stereotype.Service; import java.util.concurrent.Future; @Service @Async public class AsyncService { public Future<Student> get(){ Student stu=new Student("1","3"); try { Thread.sleep(10000l); } catch (InterruptedException e) { e.printStackTrace(); } return AsyncResult.forValue(stu); } public void executeRemote(){ try { Thread.sleep(10000l); } catch (InterruptedException e) { e.printStackTrace(); } } }
首先,要使該類讓spring管理必須使用@Service注解(或其他注解也可以),然后在類上標記@Async注解,前面說過@Async注解可以在方法或類上使用,在類上使用則表示類中的所有方法均使用異步執行的方式。異步執行類中有兩個方法,每個方法為了演示執行的耗時操作均睡眠10s。這兩個方法一個是有返回值的,另一個是無返回值的,重點看有返回值的,
public Future<Student> get(){ Student stu=new Student("1","3"); try { Thread.sleep(10000l); } catch (InterruptedException e) { e.printStackTrace(); } return AsyncResult.forValue(stu); }
為什么方法的返回值是Future,在@Async注釋上有下面這句話,
從上面的注解正好可以說明返回Future是沒問題,但是我們的方法就是一個普通的方法,要怎么才能返回Future類那,不慌,spring針對@Async注解提供了AsyncResult類,從類名就知道該類就是為了@Async注解準備的,
使用其中的forValue方法,便可以返回一個帶有泛型的Future類了。
看下測試類,
package com.example.demo.controller; import com.example.demo.Student; import com.example.demo.service.AsyncService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import java.util.concurrent.ExecutionException; import java.util.concurrent.Future; @Controller @RequestMapping("async") public class ControllerAsyncTest { @Autowired private AsyncService asyncService; @RequestMapping("/test") @ResponseBody public Student get(){ try { long start=System.currentTimeMillis(); //調用帶有返回值的get方法 Future<Student> result=asyncService.get(); //調用無返回值的executeRemote方法 asyncService.executeRemote(); Student student=result.get(); long end=System.currentTimeMillis(); System.out.println("執行時間:"+(end-start)); return student; } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } return null; } }
測試類就是一個簡單的controller,調用了get和executeRemote方法,這兩個方法分別會睡眠10s,而且get會有返回值,下面看是否可以拿到get的返回值,并看下調用這兩個方法的時間,
可以成功拿到返回值,看執行時間,
2020-12-12 21:37:43.556 INFO 11780 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet : Completed initialization in 5 ms
執行時間:10006
執行時間是10006ms,也就是10s多,按照上面的分析兩個方法分別睡眠了10s,如果同步執行那肯定是20s,把@Async注解去掉看執行時間,
2020-12-12 21:41:07.840 INFO 11584 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet : Completed initialization in 5 ms
執行時間:20001
執行時間是20001ms,算上兩個方法睡眠的時間,和測試類本身的時間,20001ms是沒錯的。從這里可以看出@Async注解的作用,把每個方法當作任務提交給了線程池,提高了任務執行的時間。
另外,在獲取異步的執行結果使用了下面的方法,
Future<Student> result=asyncService.get(); asyncService.executeRemote(); //獲得執行結果 Student student=result.get();
由于在主線程要獲得任務的執行結果,使用Future類的get方法獲得結果,該結果需要等到任務執行完以后才可以獲得。
關于springboot中的@Async怎么實現異步調用就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。