Spring中@Async注解實現(xiàn)異步調詳解
異步調用
在解釋異步調用之前,我們先來看同步調用的定義;同步就是整個處理過程順序執(zhí)行,當各個過程都執(zhí)行完畢,并返回結果。 異步調用則是只是發(fā)送了調用的指令,調用者無需等待被調用的方法完全執(zhí)行完畢,繼續(xù)執(zhí)行下面的流程。例如, 在某個調用中,需要順序調用 A, B, C三個過程方法;如他們都是同步調用,則需要將他們都順序執(zhí)行完畢之后,過程才執(zhí)行完畢; 如B為一個異步的調用方法,則在執(zhí)行完A之后,調用B,并不等待B完成,而是執(zhí)行開始調用C,待C執(zhí)行完畢之后,就意味著這個過程執(zhí)行完畢了。
概述說明
Spring中通過任務執(zhí)行器(TaskExecutor)來實現(xiàn)多線程和并發(fā)編程。使用ThreadPoolTaskExecutor可實現(xiàn)一個基于線程池的TaskExcutor。而實際開發(fā)中任務一般是異步的,我們可以在配置類中通過@EnableAsync開啟對異步任務的支持,并通過在實際執(zhí)行的Bean的方法中使用@Async注解來聲明其是一個異步任務。
從Spring3開始提供了@Async注解,該注解可以被標注在方法上,以便異步地調用該方法。調用者將在調用時先返回結果標志,方法的實際執(zhí)行將提交給Spring TaskExecutor的任務中,由指定的線程池中的線程執(zhí)行。
@Async應用自定義線程池配置類
/**
* @author 佛大Java程序員
* @since 1.0.0
*/
@Configuration
@ComponentScan("com.whl.asyncdemo")
@EnableAsync
public class TaskExecutorConfig implements AsyncConfigurer {
@Override
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();
taskExecutor.setCorePoolSize(5);
taskExecutor.setMaxPoolSize(10);
taskExecutor.setQueueCapacity(25);
taskExecutor.initialize();
return taskExecutor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return null;
}
}
說明:
* 利用@EnableAsync 注解開啟異步任務支持
* 自定義線程池需要定義的配置類中實現(xiàn)AsyncConfigurer接口并重寫getAsyncExecutor(),返回一個ThreadPoolTaskExecutor,這樣我們就獲得了一個基于線程池的TaskExecutor
執(zhí)行類
/**
* @author 佛大Java程序員
* @since 1.0.0
*/
@Service
public class AsyncTaskService {
@Async
public void executeAsyncTask(Integer i){
System.out.println("執(zhí)行異步任務1:" + i);
}
@Async
public void executeAsyncTaskPlus(Integer i){
System.out.println("執(zhí)行異步任務2 " +(i+1));
}}
說明:
@Async注解表明方法是個異步方法,如果注解在類級別,則表明該類所有的方法都是異步方法,而這里的方法自動注入使用ThreadPoolTaskExecutor作為TaskExecutor。
測試類
/**
* @author 佛大Java程序員
* @since 1.0.0
*/
public class Main {
public static void main(String[] args) {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(TaskExecutorConfig.class);
AsyncTaskService asyncTaskService = context.getBean(AsyncTaskService.class);
for (int i = 0; i < 5 ; i++) {
asyncTaskService.executeAsyncTask(i);
asyncTaskService.executeAsyncTaskPlus(i);
}
//校驗異步方法是否先返回結果值
System.out.println("返回執(zhí)行成功,驗證異步是否先返回結果值");
context.close();
}
}
執(zhí)行結果
注釋@Async

放開@Async注釋

@Async應用默認線程池
待補充
@Async調用中的事務處理機制?
在@Async標注的方法,同時也適用了@Transactional進行了標注;在其調用數(shù)據(jù)庫操作之時,將無法產(chǎn)生事務管理的控制,原因就在于其是基于異步處理的操作。那該如何給這些操作添加事務管理呢?可以將需要事務管理操作的方法放置到異步方法內部,在內部被調用的方法上添加@Transactional.
(1) 使用了@Async/@Transactional來標注,但是無法產(chǎn)生事務控制的目的。
(2) 使用了@Async來標注, B中調用了C、D,C/D分別使用@Transactional做了標注,則可實現(xiàn)事務控制的目的。
項目實戰(zhàn)
WKD項目里面使用Easypoi技術來導出Excel,導出數(shù)據(jù)量數(shù)據(jù)量不大的時候用同步導出也沒事,但是對于數(shù)據(jù)量大且需要導出的數(shù)據(jù)封裝業(yè)務較復雜,就會出現(xiàn)應用OOM或者出現(xiàn)響應超時。為了解決此問題,通過用線程池異步導出的方式實現(xiàn)。

到此這篇關于Spring中@Async注解實現(xiàn)異步調詳解的文章就介紹到這了,更多相關Spring @Async注解異步調用內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
Fluent MyBatis實現(xiàn)動態(tài)SQL
MyBatis 令人喜歡的一大特性就是動態(tài) SQL。本文主要介紹了Fluent MyBatis實現(xiàn)動態(tài)SQL,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-08-08
mybatis中關于type-aliases-package的使用
這篇文章主要介紹了mybatis中關于type-aliases-package的使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08
java編程實現(xiàn)求質數(shù)與因式分解代碼分享
這篇文章主要介紹了Java編程實現(xiàn)求質數(shù)與因式分解代碼分享,對二者的概念作了簡單介紹(多此一舉,哈哈),都是小學數(shù)學老師的任務,然后分享了求解質數(shù)和因式分解的Java代碼,具有一定借鑒價值,需要的朋友可以參考下。2017-12-12
java多線程之線程,進程和Synchronized概念初解
這篇文章主要介紹了java多線程之線程,進程和Synchronized概念初解,涉及進程與線程的簡單概念,實現(xiàn)多線程的方式,線程安全問題,synchronized修飾符等相關內容,具有一定借鑒價值,需要的朋友可以參考下。2017-11-11

