0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

阿里巴巴為什么不建議直接使用@Async注解?

Android編程精選 ? 來源:碼農(nóng)架構(gòu) ? 作者:碼農(nóng)架構(gòu) ? 2022-10-26 11:32 ? 次閱讀

序言

67866878-5466-11ed-a3b6-dac502259ad0.png

對于異步方法調(diào)用,從Spring3開始提供了@Async注解,該注解可以被標(biāo)在方法上,以便異步地調(diào)用該方法。調(diào)用者將在調(diào)用時立即返回,方法的實際執(zhí)行將提交給Spring TaskExecutor的任務(wù)中,由指定的線程池中的線程執(zhí)行。

在項目應(yīng)用中,@Async調(diào)用線程池,推薦使用自定義線程池的模式。自定義線程池常用方案:重新實現(xiàn)接口AsyncConfigurer。

應(yīng)用場景

同步:同步就是整個處理過程順序執(zhí)行,當(dāng)各個過程都執(zhí)行完畢,并返回結(jié)果。

異步:異步調(diào)用則是只是發(fā)送了調(diào)用的指令,調(diào)用者無需等待被調(diào)用的方法完全執(zhí)行完畢;而是繼續(xù)執(zhí)行下面的流程。例如, 在某個調(diào)用中,需要順序調(diào)用 A, B, C三個過程方法;如他們都是同步調(diào)用,則需要將他們都順序執(zhí)行完畢之后,方算作過程執(zhí)行完畢;如B為一個異步的調(diào)用方法,則在執(zhí)行完A之后,調(diào)用B,并不等待B完成,而是執(zhí)行開始調(diào)用C,待C執(zhí)行完畢之后,就意味著這個過程執(zhí)行完畢了。在Java中,一般在處理類似的場景之時,都是基于創(chuàng)建獨立的線程去完成相應(yīng)的異步調(diào)用邏輯,通過主線程和不同的業(yè)務(wù)子線程之間的執(zhí)行流程,從而在啟動獨立的線程之后,主線程繼續(xù)執(zhí)行而不會產(chǎn)生停滯等待的情況。

Spring已經(jīng)實現(xiàn)的線程池

SimpleAsyncTaskExecutor:不是真的線程池,這個類不重用線程,默認(rèn)每次調(diào)用都會創(chuàng)建一個新的線程。

SyncTaskExecutor:這個類沒有實現(xiàn)異步調(diào)用,只是一個同步操作。只適用于不需要多線程的地方。

ConcurrentTaskExecutor:Executor的適配類,不推薦使用。如果ThreadPoolTaskExecutor不滿足要求時,才用考慮使用這個類。

SimpleThreadPoolTaskExecutor:是Quartz的SimpleThreadPool的類。線程池同時被quartz和非quartz使用,才需要使用此類。

ThreadPoolTaskExecutor :最常使用,推薦。其實質(zhì)是對java.util.concurrent.ThreadPoolExecutor的包裝。

異步的方法有:

最簡單的異步調(diào)用,返回值為void

參數(shù)的異步調(diào)用,異步方法可以傳入?yún)?shù)

存在返回值,常調(diào)用返回Future

Spring中啟用@Async

67efdcae-5466-11ed-a3b6-dac502259ad0.jpg ?

@Async應(yīng)用默認(rèn)線程池

Spring應(yīng)用默認(rèn)的線程池,指在@Async注解在使用時,不指定線程池的名稱。查看源碼,@Async的默認(rèn)線程池為SimpleAsyncTaskExecutor。

無返回值調(diào)用

基于@Async無返回值調(diào)用,直接在使用類,使用方法(建議在使用方法)上,加上注解。若需要拋出異常,需手動new一個異常拋出。
681adc56-5466-11ed-a3b6-dac502259ad0.jpg

有返回值Future調(diào)用

686b150e-5466-11ed-a3b6-dac502259ad0.jpg

有返回值CompletableFuture調(diào)用

CompletableFuture 并不使用@Async注解,可達(dá)到調(diào)用系統(tǒng)線程池處理業(yè)務(wù)的功能。 JDK5新增了Future接口,用于描述一個異步計算的結(jié)果。雖然 Future 以及相關(guān)使用方法提供了異步執(zhí)行任務(wù)的能力,但是對于結(jié)果的獲取卻是很不方便,只能通過阻塞或者輪詢的方式得到任務(wù)的結(jié)果。阻塞的方式顯然和我們的異步編程的初衷相違背,輪詢的方式又會耗費無謂的 CPU 資源,而且也不能及時地得到計算結(jié)果。

CompletionStage代表異步計算過程中的某一個階段,一個階段完成以后可能會觸發(fā)另外一個階段

一個階段的計算執(zhí)行可以是一個Function,Consumer或者Runnable。比如:

stage.thenApply(x -> square(x)).thenAccept(x -> System.out.print(x)).thenRun(() -> System.out.println())

一個階段的執(zhí)行可能是被單個階段的完成觸發(fā),也可能是由多個階段一起觸發(fā)

在Java8中,CompletableFuture 提供了非常強大的Future的擴展功能,可以幫助我們簡化異步編程的復(fù)雜性,并且提供了函數(shù)式編程的能力,可以通過回調(diào)的方式處理計算結(jié)果,也提供了轉(zhuǎn)換和組合 CompletableFuture 的方法。

它可能代表一個明確完成的Future,也有可能代表一個完成階段( CompletionStage ),它支持在計算完成以后觸發(fā)一些函數(shù)或執(zhí)行某些動作。

它實現(xiàn)了Future和CompletionStage接口

689357d0-5466-11ed-a3b6-dac502259ad0.jpg

默認(rèn)線程池的弊端

在線程池應(yīng)用中,參考阿里巴巴java開發(fā)規(guī)范:線程池不允許使用Executors去創(chuàng)建,不允許使用系統(tǒng)默認(rèn)的線程池,推薦通過ThreadPoolExecutor的方式,這樣的處理方式讓開發(fā)的工程師更加明確線程池的運行規(guī)則,規(guī)避資源耗盡的風(fēng)險。

Executors各個方法的弊端:

newFixedThreadPool和newSingleThreadExecutor:主要問題是堆積的請求處理隊列可能會耗費非常大的內(nèi)存,甚至OOM。

newCachedThreadPool和newScheduledThreadPool:要問題是線程數(shù)最大數(shù)是Integer.MAX_VALUE,可能會創(chuàng)建數(shù)量非常多的線程,甚至OOM。

@Async默認(rèn)異步配置使用的是SimpleAsyncTaskExecutor,該線程池默認(rèn)來一個任務(wù)創(chuàng)建一個線程,若系統(tǒng)中不斷的創(chuàng)建線程,最終會導(dǎo)致系統(tǒng)占用內(nèi)存過高,引發(fā)OutOfMemoryError錯誤。針對線程創(chuàng)建問題,SimpleAsyncTaskExecutor提供了限流機制,通過concurrencyLimit屬性來控制開關(guān),當(dāng)concurrencyLimit>=0時開啟限流機制,默認(rèn)關(guān)閉限流機制即concurrencyLimit=-1,當(dāng)關(guān)閉情況下,會不斷創(chuàng)建新的線程來處理任務(wù)?;谀J(rèn)配置,SimpleAsyncTaskExecutor并不是嚴(yán)格意義的線程池,達(dá)不到線程復(fù)用的功能。歡迎關(guān)注公眾號"Java學(xué)習(xí)之道",查看更多干貨!

@Async應(yīng)用自定義線程池

自定義線程池,可對系統(tǒng)中線程池更加細(xì)粒度的控制,方便調(diào)整線程池大小配置,線程執(zhí)行異??刂坪吞幚?。在設(shè)置系統(tǒng)自定義線程池代替默認(rèn)線程池時,雖可通過多種模式設(shè)置,但替換默認(rèn)線程池最終產(chǎn)生的線程池有且只能設(shè)置一個(不能設(shè)置多個類繼承AsyncConfigurer)自定義線程池有如下模式:

重新實現(xiàn)接口AsyncConfigurer

繼承AsyncConfigurerSupport

配置由自定義的TaskExecutor替代內(nèi)置的任務(wù)執(zhí)行器

通過查看Spring源碼關(guān)于@Async的默認(rèn)調(diào)用規(guī)則,會優(yōu)先查詢源碼中實現(xiàn)AsyncConfigurer這個接口的類,實現(xiàn)這個接口的類為AsyncConfigurerSupport。但默認(rèn)配置的線程池和異步處理方法均為空,所以,無論是繼承或者重新實現(xiàn)接口,都需指定一個線程池。且重新實現(xiàn) public Executor getAsyncExecutor()方法。

實現(xiàn)接口AsyncConfigurer

68c7f602-5466-11ed-a3b6-dac502259ad0.jpg

繼承AsyncConfigurerSupport

69311c2c-5466-11ed-a3b6-dac502259ad0.jpg

配置自定義的TaskExecutor

由于AsyncConfigurer的默認(rèn)線程池在源碼中為空,Spring通過beanFactory.getBean(TaskExecutor.class)先查看是否有線程池,未配置時,通過beanFactory.getBean(DEFAULT_TASK_EXECUTOR_BEAN_NAME, Executor.class),又查詢是否存在默認(rèn)名稱為TaskExecutor的線程池。所以可在項目中,定義名稱為TaskExecutor的bean生成一個默認(rèn)線程池。也可不指定線程池的名稱,申明一個線程池,本身底層是基于TaskExecutor.class便可。 比如:

Executor.class:ThreadPoolExecutorAdapter->ThreadPoolExecutor->AbstractExecutorService->ExecutorService->Executor
這樣的模式,最終底層為Executor.class,在替換默認(rèn)的線程池時,需設(shè)置默認(rèn)的線程池名稱為TaskExecutor
TaskExecutor.class:ThreadPoolTaskExecutor->SchedulingTaskExecutor->AsyncTaskExecutor->TaskExecutor
這樣的模式,最終底層為TaskExecutor.class,在替換默認(rèn)的線程池時,可不指定線程池名稱。 6973833c-5466-11ed-a3b6-dac502259ad0.jpg ?

審核編輯:湯梓紅
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • spring
    +關(guān)注

    關(guān)注

    0

    文章

    340

    瀏覽量

    14353
  • 線程池
    +關(guān)注

    關(guān)注

    0

    文章

    57

    瀏覽量

    6853
  • 注解
    +關(guān)注

    關(guān)注

    0

    文章

    18

    瀏覽量

    2683

原文標(biāo)題:阿里巴巴為什么不建議直接使用@Async注解?

文章出處:【微信號:AndroidPush,微信公眾號:Android編程精選】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    榮邦佳業(yè)有阿里巴巴

    了吧。那就是阿里巴巴。阿里巴巴成立一周年之際,通過馬云的積極奔走并請大俠金庸從中穿針引線、廣發(fā)英雄帖,中國互聯(lián)網(wǎng)的風(fēng)云人物---新浪的王志東、網(wǎng)易的丁磊、搜狐的張朝陽以及8848的王峻濤等,在馬云的誠意
    發(fā)表于 04-15 10:04

    阿里巴巴Android開發(fā)手冊》正式發(fā)布,獻給移動開發(fā)者的新年禮物

    (《阿里巴巴Java開發(fā)手冊》主要作者)對手冊的指導(dǎo)。最后越早遵循統(tǒng)一的開發(fā)規(guī)范,團隊就越受益,希望該手冊能夠為Android開發(fā)團隊和開發(fā)者帶來切實的幫助,同時,我們也呼吁廣大Android開發(fā)者參與進來,不斷完善該規(guī)范,您可以在評論處提交您的建議。識別以下二維碼,閱讀
    發(fā)表于 02-28 16:29

    阿里巴巴敏捷研發(fā)的探索與實踐

    摘要: 今天你敏捷了嗎?敏捷產(chǎn)品開發(fā)提倡快速迭代、小步快跑,以便更靈活地應(yīng)對變化,目前逐漸演變?yōu)樾袠I(yè)潮流。阿里巴巴內(nèi)部也在不斷進行敏捷實踐。點此查看原文:http
    發(fā)表于 03-07 17:26

    阿里巴巴開源的通用緩存訪問框架JetCache介紹

    Cache的注解式緩存,支持TTL、多級緩存、分布式自動刷新,也提供類似JSR107規(guī)范的Cache API。JetCache是由阿里巴巴開源的通用緩存訪問框架,如果你對Spring Cache很熟悉
    發(fā)表于 04-24 16:09

    國內(nèi)芯片—阿里巴巴的人工智能實力

    成立于2016年,2017年首次對外公布的阿里巴巴工智能實驗室(A.I. Labs)目前主要聚焦AI技術(shù),研究對象包括機器學(xué)習(xí)、下一代人機交互、自動駕駛、服務(wù)機器人等領(lǐng)域。 云棲大會上,阿里巴巴
    發(fā)表于 09-27 17:54

    阿里巴巴禁止開發(fā)人員使用isSuccess作為變量名的原因是什么

    為什么阿里巴巴禁止開發(fā)人員使用isSuccess作為變量名
    發(fā)表于 04-08 14:46

    詳解阿里巴巴智能對話開發(fā)平臺

    四大維度全景揭秘阿里巴巴智能對話開發(fā)平臺
    發(fā)表于 07-31 11:15

    哪里可以下載阿里巴巴的地方和鏈接?

    你知道我可以從哪里下載阿里巴巴的地方和鏈接嗎?
    發(fā)表于 11-04 09:25

    阿里巴巴無線空口提案正式成為LoRa國際標(biāo)準(zhǔn)

    技術(shù)之一,已經(jīng)在全世界各個國家得到了廣泛應(yīng)用,無線頻譜技術(shù)標(biāo)準(zhǔn)是無線通信領(lǐng)域的“重中之重”,直接影響網(wǎng)絡(luò)的性能、競爭力和建設(shè)成本。對此,阿里巴巴提出了《LoRaWAN CN470頻譜技術(shù)標(biāo)準(zhǔn)》方案,并經(jīng)
    發(fā)表于 12-13 10:10

    數(shù)據(jù)揭秘你不知道的阿里巴巴數(shù)據(jù)中心

    Alibaba Cluster Data 開源:270GB 數(shù)據(jù)揭秘你不知道的阿里巴巴數(shù)據(jù)中心
    發(fā)表于 04-30 09:37

    阿里巴巴ai模型破世界紀(jì)錄_阿里巴巴的人工智能未來發(fā)展解析

    阿里巴巴近日再爆大事件,阿里巴巴ai模型在閱讀理解領(lǐng)域頂級賽事SQuAD破世界紀(jì)錄引起大家的高度關(guān)注,跟隨小編一起來了解一下是什么情況,看看阿里巴巴人工智能的未來發(fā)展趨勢以及全球人工智能的未來發(fā)展趨勢是什么。
    發(fā)表于 01-16 11:47 ?1718次閱讀

    阿里巴巴回應(yīng)“收縮招聘”:系假新聞,只是阿里巴巴近期的一次人才盤點和人才體系升級

    速途網(wǎng)10月23日消息(報道:吳佳馨)近日有傳言稱,阿里巴巴已縮減headcount,所有業(yè)務(wù)線批offer。今日下午,阿里巴巴對此傳言做出回應(yīng),稱該消息為假新聞,這只是阿里巴巴近期
    發(fā)表于 10-25 11:15 ?413次閱讀

    阿里巴巴馬云占股份多少?阿里巴巴馬云持股比例?目前馬云持股降至4.8%

    馬云在阿里巴巴占多少股份你知道嗎?之前大概馬云在阿里巴巴的持股比例為7%,現(xiàn)在馬云在阿里持股降至5%以下但是仍為最大個人股東。
    發(fā)表于 07-13 17:02 ?2.6w次閱讀
    <b class='flag-5'>阿里巴巴</b>馬云占股份多少?<b class='flag-5'>阿里巴巴</b>馬云持股比例?目前馬云持股降至4.8%

    阿里巴巴注冊元宇宙

    近日,阿里巴巴元宇宙商標(biāo)已經(jīng)被其他公司搶注,目前商標(biāo)狀態(tài)均為申請中。阿里巴巴通過成立XR 實驗室逐步實現(xiàn)元宇宙的四個階段布局,擁抱元宇宙。
    的頭像 發(fā)表于 11-08 11:40 ?3189次閱讀

    阿里巴巴完成雙重主要上市

    8月28日阿里巴巴在香港聯(lián)交所主板主要上市,成為在港交所、紐交所雙重主要上市的公司。如今,阿里巴巴完成雙重主要上市。 此外,有投資機構(gòu)認(rèn)為阿里巴巴最快有望于9月份就納入港股通。 截至8月28日收盤,
    的頭像 發(fā)表于 08-29 16:06 ?378次閱讀