SpringBoot項(xiàng)目@Async方法問題解決方案
現(xiàn)象:
1. 表面現(xiàn)象: 方法中輸出的日志, 日志文件中找不到, 也沒有任何報(bào)錯(cuò)(即@Async標(biāo)注的方法沒有執(zhí)行, 也沒有報(bào)錯(cuò))
2. 分析現(xiàn)象: 日志中某個(gè)時(shí)刻之后沒有了task-xxx線程的日志
原因:
@Async異常方法默認(rèn)使用Spring創(chuàng)建ThreadPoolTaskExecutor(參考TaskExecutionAutoConfiguration),
其中默認(rèn)核心線程數(shù)為8, 默認(rèn)最大隊(duì)列和默認(rèn)最大線程數(shù)都是Integer.MAX_VALUE. 創(chuàng)建新線程的條件是隊(duì)列填滿時(shí), 而這樣的配置隊(duì)列永遠(yuǎn)不會(huì)填滿, 如果有@Async注解標(biāo)注的方法長期占用線程(比如HTTP長連接等待獲取結(jié)果), 在核心8個(gè)線程數(shù)占用滿了之后, 新的調(diào)用就會(huì)進(jìn)入隊(duì)列, 外部表現(xiàn)為沒有執(zhí)行.
解決:
手動(dòng)配置相應(yīng)屬性即可. 比如
spring.task.execution.pool.queueCapacity=4spring.task.execution.pool.coreSize=20
備注:
此處沒有配置maxSize, 仍是默認(rèn)的Integer.MAX_VALUE. 如果配置的話, 請考慮達(dá)到最大線程數(shù)時(shí)的處理策略(JUC包查找RejectedExecutionHandler的實(shí)現(xiàn)類)
(默認(rèn)為拒絕執(zhí)行AbortPolicy, 即拋出異常)
AbortPolicy: 直接拋出java.util.concurrent.RejectedExecutionException異常
CallerRunsPolicy: 主線程直接執(zhí)行該任務(wù),執(zhí)行完之后嘗試添加下一個(gè)任務(wù)到線程池中,可以有效降低向線程池內(nèi)添加任務(wù)的速度
DiscardOldestPolicy: 拋棄舊的任務(wù)
DiscardPolicy: 拋棄當(dāng)前任務(wù)
截圖:
1. ThreadPoolTaskExecutor
2. SpringMonitor的配置屬性
3. SpringMonitor的Threads
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. Python如何實(shí)現(xiàn)感知器的邏輯電路2. Python getsizeof()和getsize()區(qū)分詳解3. JS實(shí)現(xiàn)表單中點(diǎn)擊小眼睛顯示隱藏密碼框中的密碼4. vue實(shí)現(xiàn)移動(dòng)端返回頂部5. asp讀取xml文件和記數(shù)6. python基于scrapy爬取京東筆記本電腦數(shù)據(jù)并進(jìn)行簡單處理和分析7. 原生js實(shí)現(xiàn)的觀察者和訂閱者模式簡單示例8. Python ellipsis 的用法詳解9. 在終端啟動(dòng)Python時(shí)報(bào)錯(cuò)的解決方案10. JS錯(cuò)誤處理與調(diào)試操作實(shí)例分析
