国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

Spring之異步任務(wù)@Async詳解分析

這篇具有很好參考價值的文章主要介紹了Spring之異步任務(wù)@Async詳解分析。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。

1 異步@Async詳解

1.1 引言

java中異步線程很重要,比如在業(yè)務(wù)流處理時,需要通知硬件設(shè)備,發(fā)短信通知用戶,或者需要上傳一些圖片資源到其他服務(wù)器這種耗時的操作,在主線程里處理會阻塞整理流程,而且我們也不需要等待處理結(jié)果之后再進行下一步操作,這時候就可以使用異步線程進行處理,這樣主線程不會因為這些耗時的操作而阻塞,保證主線程的流程可以正常進行。
最近在項目中使用了很多線程的操作,在這做個記錄

1.2 異步說明和原理

使用地方說明:

  • 在方法上使用該@Async注解,申明該方法是一個異步任務(wù);
  • 在類上面使用該@Async注解,申明該類中的所有方法都是異步任務(wù);
  • 使用此注解的方法的類對象,必須是spring管理下的bean對象;
  • 要想使用異步任務(wù),需要在主類上開啟異步配置,即,配置上@EnableAsync注解;

@Async的原理概括:
@Async的原理是通過 Spring AOP 動態(tài)代理 的方式來實現(xiàn)的。
Spring 容器啟動初始化bean時,判斷類中是否使用了@Async注解,如果使用了則為其創(chuàng)建切入點和切入點處理器,根據(jù)切入點創(chuàng)建代理,在線程調(diào)用@Async注解標注的方法時,會調(diào)用代理,執(zhí)行切入點處理器invoke方法,將方法的執(zhí)行提交給線程池中的另外一個線程來處理,從而實現(xiàn)了異步執(zhí)行。
所以,需要注意的一個錯誤用法是,如果a方法調(diào)用它同類中的標注@Asyncb方法,是不會異步執(zhí)行的,因為從a方法進入調(diào)用的都是該類對象本身,不會進入代理類。
因此,相同類中的方法調(diào)用帶@Async的方法是無法異步的,這種情況仍然是同步。

1.3 @Async使用

Spring中啟用@Async

  • @Async注解在使用時,如果不指定線程池的名稱,則使用Spring默認的線程池,Spring默認的線程池為SimpleAsyncTaskExecutor。
  • 方法上一旦標記了這個@Async注解,當其它線程調(diào)用這個方法時,就會開啟一個新的子線程去異步處理該業(yè)務(wù)邏輯。

1.3.1 啟動類中增加@EnableAsync

Spring boot為例,啟動類中增加@EnableAsync

@EnableAsync
@SpringBootApplication
public class ManageApplication {
    //...
}

1.3.2 方法上加@Async注解

@Component
public class MyAsyncTask {
     @Async
    public void asyncCpsItemImportTask(Long platformId, String jsonList){
        //...具體業(yè)務(wù)邏輯
    }
}

1.4 @Async異步線程池

1.4.1 默認線程池

上面的配置會啟用默認的線程池/執(zhí)行器,異步執(zhí)行指定的方法。

Spring默認的線程池的默認配置:

默認核心線程數(shù):8,
最大線程數(shù):Integet.MAX_VALUE,
隊列使用LinkedBlockingQueue,
容量是:Integet.MAX_VALUE,
空閑線程保留時間:60s,
線程池拒絕策略:AbortPolicy

缺點:從最大線程數(shù)的配置上,相信看到問題:并發(fā)情況下,會無限創(chuàng)建線程

默認線程池的上述缺陷如何解決:答案是,自定義配置參數(shù)就可以了

1.4.3 在配置文件中配置

spring:
  task:
    execution:
      pool:
        max-size: 6
        core-size: 3
        keep-alive: 3s
        queue-capacity: 1000
        thread-name-prefix: name

1.4.3 自定義線程池

在業(yè)務(wù)場景中,有時需要使用自己定義的執(zhí)行器來跑異步的業(yè)務(wù)邏輯,那該怎么辦呢?答案是,自定義線程池。

1.4.3.1 編寫配置類
@Configuration
@Data
public class ExecutorConfig{
    //核心線程
    private int corePoolSize;
    //最大線程
    private int maxPoolSize;
    //隊列容量
    private int queueCapacity;
    //保持時間
    private int keepAliveSeconds;
    //名稱前綴
    private String preFix;
 
    @Bean("MyExecutor")
    public Executor myExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(corePoolSize);
        executor.setMaxPoolSize(maxPoolSize);
        executor.setQueueCapacity(queueCapacity);
        executor.setKeepAliveSeconds(keepAliveSeconds);
        executor.setThreadNamePrefix(preFix);
        executor.setRejectedExecutionHandler( new ThreadPoolExecutor.AbortPolicy());
        executor.initialize();
        return executor;
    }
}
1.4.3.2 使用自定義線程池

在方法上的@Async注解處指定線程池名字:

@Component
public class MyAsyncTask {
     @Async("MyExecutor") //使用自定義的線程池(執(zhí)行器)
    public void asyncCpsItemImportTask(Long platformId, String jsonList){
        //...具體業(yè)務(wù)邏輯
    }
}

1.4.4 Spring中的線程池(執(zhí)行器)

SpringTaskExecutorTaskScheduler接口提供了異步執(zhí)行調(diào)度任務(wù)的抽象。

SpringTaskExecutorjava.util.concurrent.Executor接口時一樣的,這個接口只有一個方法execute(Runnable task)

Spring已經(jīng)內(nèi)置了許多TaskExecutor的實現(xiàn),沒有必要自己去實現(xiàn):

  • SimpleAsyncTaskExecutor: 這種實現(xiàn)不會重用任何線程,每次調(diào)用都會創(chuàng)建一個新的線程。
  • SyncTaskExecutor: 這種實現(xiàn)不會異步的執(zhí)行,相反,每次調(diào)用都在發(fā)起調(diào)用的線程中執(zhí)行。它的主要用處是在不需要多線程的時候,比如簡單的測試用例;
  • ConcurrentTaskExecutor:這個實現(xiàn)是對Java 5 java.util.concurrent.Executor類的包裝。有另一個ThreadPoolTaskExecutor類更為好用,它暴露了Executor的配置參數(shù)作為bean屬性。
    點擊了解Spring線程池ThreadPoolTaskExecutor講解
  • SimpleThreadPoolTaskExecutor: 這個實現(xiàn)實際上是QuartzSimpleThreadPool類的子類,它會監(jiān)聽Spring的生命周期回調(diào)。當有線程池,需要在Quartz非Quartz組件中共用時,這是它的典型用處。
  • ThreadPoolTaskExecutor:這是最常用、最通用的一種實現(xiàn)。它包含了java.util.concurrent.ThreadPoolExecutor的屬性,并且用TaskExecutor進行包裝。

1.5 異步中的事務(wù)和返回

1.5.1 異步事務(wù)

@Async標注的方法,同時也使用@Transactional進行標注;在其調(diào)用數(shù)據(jù)庫操作之時,將無法產(chǎn)生事務(wù)管理的控制,原因就在于其是基于異步處理的操作。
那該如何給這些操作添加事務(wù)管理呢?
可以將需要事務(wù)管理操作的方法放置到異步方法內(nèi)部,在內(nèi)部被調(diào)用的方法上添加@Transactional

示例:

  • 方法A:使用了@Async/@Transactional來標注,但是無法產(chǎn)生事務(wù)控制的目的。
  • 方法B:使用了@Async來標注,B中調(diào)用了C、D,C/D分別使用@Transactional做了標注,則可實現(xiàn)事務(wù)控制的目的

1.5.2 異步返回

異步的業(yè)務(wù)邏輯處理場景 有兩種:一個是不需要返回結(jié)果,另一種是需要接收返回結(jié)果。不需要返回結(jié)果的比較簡單,就不多說了。
需要接收返回結(jié)果的示例如下:

@Async("MyExecutor")
public Future<Map<Long, List>> queryMap(List ids) {
    List<> result = businessService.queryMap(ids);
    ..............
    Map<Long, List> resultMap = Maps.newHashMap();
    ...
    return new AsyncResult<>(resultMap);
}

調(diào)用異步方法的示例:

public Map<Long, List> asyncProcess(List<BindDeviceDO> bindDevices,List<BindStaffDO> bindStaffs, String dccId) {
      Map<Long, List> finalMap =null;
      // 返回值:
      Future<Map<Long, List>> asyncResult = MyService.queryMap(ids);
      try {
          finalMap = asyncResult.get();
      } catch (Exception e) {
          ...
      }
      return finalMap;
}

1.6 異步不能回調(diào)問題

使用了異步但是執(zhí)行異步的方法,原因是在方法上加了@Async注解,之所以加這個注解是因為報錯:

There was an unexpected error (type=Internal Server Error, status=500).
Async support must be enabled on a servlet and for all filters involved in async request processing. This is done in Java code using the Servlet API or by adding <async-supported>true</async-supported> to servlet and filter declarations in web.xml

異步測試時一直報這個錯誤,提示我在web.xml開啟異步支持,但是我是SpringBoot項目,于是開始網(wǎng)上查找
錯誤:加@Async注解,會更加異步,不能獲取異步結(jié)果
正確:根本原因是容器注冊問題,在springboot啟動類的注解@SpringBootApplication旁邊添加了@ServletComponentScan,才導致上面的報錯和不能回調(diào),有三種解決方法:

  • 去掉注解@ServletComponentScan
  • 添加容器注冊(springboot項目)
  @Bean
    public ServletRegistrationBean dispatcherServlet() {
        ServletRegistrationBean registration = new ServletRegistrationBean(
                new DispatcherServlet(), "/");
        registration.setAsyncSupported(true);
        return registration;
    }
    @Bean
    DispatcherServlet dispatcherServlet(){
        return new DispatcherServlet();
    }

在過濾器那里添加asyncSupported = true的支持文章來源地址http://www.zghlxwxcb.cn/news/detail-430290.html

@WebFilter(urlPatterns="/*",asyncSupported = true)
  • 修改web.xml(傳統(tǒng)xml項目)
    需要在 web.xml 文件中的 servlet 定義中添加:"<async-supported>true</async-supported>"
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns="http://java.sun.com/xml/ns/javaee"
	xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
	version="3.0">
	<display-name>Archetype Created Web Application</display-name>
	<context-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>classpath:spring-mybatis.xml</param-value>
	</context-param>
	<context-param>
		<param-name>spring.profiles.active</param-name>
		<param-value>dev</param-value>
	</context-param>
	<context-param>
		<param-name>spring.profiles.default</param-name>
		<param-value>dev</param-value>
	</context-param>
	<filter>
		<filter-name>encodingFilter</filter-name>
		<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
		<async-supported>true</async-supported>
		<init-param>
			<param-name>encoding</param-name>
			<param-value>UTF-8</param-value>
		</init-param>
		<init-param>
			<param-name>forceEncoding</param-name>
			<param-value>true</param-value>
		</init-param>
	</filter>
	<filter-mapping>
		<filter-name>encodingFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>

	<listener>
		<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
	</listener>

	<servlet>
		<servlet-name>SpringMVC</servlet-name>
		<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
		<init-param>
			<param-name>contextConfigLocation</param-name>
			<param-value>classpath:spring-mvc.xml</param-value>
		</init-param>
		<load-on-startup>1</load-on-startup>
		<async-supported>true</async-supported>
	</servlet>
	<servlet-mapping>
		<servlet-name>SpringMVC</servlet-name>
		<url-pattern>/</url-pattern>
	</servlet-mapping>
</web-app>

到了這里,關(guān)于Spring之異步任務(wù)@Async詳解分析的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務(wù),不擁有所有權(quán),不承擔相關(guān)法律責任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實不符,請點擊違法舉報進行投訴反饋,一經(jīng)查實,立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費用

相關(guān)文章

  • Ajax_4(進階)同步異步+ 宏任務(wù)微任務(wù) + Promise鏈 + async終極解決方案 +事件循環(huán)原理 + 綜合案例

    01-同步代碼和異步代碼 什么是同步代碼? 同步代碼:逐行執(zhí)行,需要原地等待結(jié)果后,才繼續(xù)向下執(zhí)行。 什么是異步代碼? 調(diào)用后耗時,不阻塞代碼繼續(xù)執(zhí)行,(不必原地等待),在將來完成后 觸發(fā)一個 回調(diào)函數(shù) 。 代碼閱讀 目標:閱讀并回答代碼執(zhí)行和打印的順序 打印

    2024年02月13日
    瀏覽(51)
  • Swift async/await 并發(fā)中如何將任務(wù)組(TaskGroup)轉(zhuǎn)換為異步序列(AsyncSequence)

    Swift async/await 并發(fā)中如何將任務(wù)組(TaskGroup)轉(zhuǎn)換為異步序列(AsyncSequence)

    在 Swift 新結(jié)構(gòu)化并行模型的開發(fā)中,? 提出了任務(wù)組(TaskGroup)和異步序列(AsyncSequence)的概念。有時候,為了簡潔和效率方面的原因,我們需要將 TaskGroup 的結(jié)果轉(zhuǎn)換為異步序列。 如上圖所示,我們試圖將任務(wù)組的結(jié)果轉(zhuǎn)換為異步序列,但不幸失敗了。 那么,最終我們到

    2024年02月12日
    瀏覽(23)
  • async和await用法理解和快速上手 , 同步任務(wù)和異步任務(wù)順序安排和輕松理解 , js代碼執(zhí)行順序表面知道

    async和await用法理解和快速上手 , 同步任務(wù)和異步任務(wù)順序安排和輕松理解 , js代碼執(zhí)行順序表面知道

    學習關(guān)鍵語句 : async , await 用法 await 怎么使用 同步任務(wù)和異步任務(wù) 微任務(wù)和宏任務(wù) js中代碼執(zhí)行順序 雖然說 async 和 await 是 Promise 的語法糖 , 但是用慣了Promise 的人(我) , 還真不能超快速使用上這個語法糖 , 所以趕緊寫一篇文章出來讓各位了解了解這個到底怎么用在我的項目

    2024年02月03日
    瀏覽(48)
  • 深入理解Spring的@Async注解:實現(xiàn)異步方法調(diào)用

    在當今高速發(fā)展的應(yīng)用開發(fā)領(lǐng)域,對于提升系統(tǒng)性能和響應(yīng)能力的需求越來越迫切。而異步編程作為一種解決方案,已經(jīng)成為現(xiàn)代應(yīng)用開發(fā)中的一項重要技術(shù)。本篇博客將帶您深入探究 Java 中的 @Async 注解,揭示其強大的異步執(zhí)行能力和精妙的實現(xiàn)機制。 異步編程是一種編程

    2024年02月05日
    瀏覽(26)
  • Spring Boot使用@Async實現(xiàn)異步調(diào)用:自定義線程池

    第一步,先在Spring Boot主類中定義一個線程池,比如: 上面我們通過使用?? ThreadPoolTaskExecutor ??創(chuàng)建了一個線程池,同時設(shè)置了以下這些參數(shù): 核心線程數(shù)10:線程池創(chuàng)建時候初始化的線程數(shù) 最大線程數(shù)20:線程池最大的線程數(shù),只有在緩沖隊列滿了之后才會申請超過核

    2024年02月14日
    瀏覽(26)
  • java小技能:spring中的異步方法@Async失效的原因

    異步執(zhí)行的場景:不處理方法結(jié)果/在不關(guān)心方法執(zhí)行結(jié)果時經(jīng)常需要異步執(zhí)行 需求:自動報名活動、批量導入報名活動

    2024年02月09日
    瀏覽(21)
  • 【js】js 異步機制詳解 Generator / Async / Promise

    【js】js 異步機制詳解 Generator / Async / Promise

    三種語法功能放在一起,是因為他們都有相似特點: 維護某種狀態(tài) 在未來恢復狀態(tài)并執(zhí)行 本文重點回答以下幾個問題: 為什么 Generator 和 Async 函數(shù)的 代碼執(zhí)行流 都可以簡化成樹形結(jié)構(gòu)? async 函數(shù)為什么返回一個 promise?返回了怎樣一個 promise? async 函數(shù)如何優(yōu)雅的轉(zhuǎn)換成

    2024年01月21日
    瀏覽(20)
  • Winform中使用HttpClient(設(shè)置最大超時響應(yīng)時間)調(diào)用接口并做業(yè)務(wù)處理時界面卡住,使用async Task await異步任務(wù)編程優(yōu)化

    Winform中使用HttpClient(設(shè)置最大超時響應(yīng)時間)調(diào)用接口并做業(yè)務(wù)處理時界面卡住,使用async Task await異步任務(wù)編程優(yōu)化

    Winform中怎樣使用HttpClient調(diào)用http的get和post接口并將接口返回json數(shù)據(jù)解析為實體類: Winform中怎樣使用HttpClient調(diào)用http的get和post接口并將接口返回json數(shù)據(jù)解析為實體類_winform請求http接口_霸道流氓氣質(zhì)的博客-CSDN博客 參考前面使用HttpClient調(diào)用http的get和post接口的小示例, 需要定

    2024年02月02日
    瀏覽(71)
  • 詳解async 與 await,帶您理解Playwright使用異步方法的正確姿勢!

    詳解async 與 await,帶您理解Playwright使用異步方法的正確姿勢!

    大家在使用python做playwright自動化測試的過程中,一定會發(fā)現(xiàn)下面這種異步用法 很多同學可能只是按照這種寫法來編寫項目的自動化測試代碼,對于具體細節(jié)可能并不了解,今天我就來講一下playwright異步用法的相關(guān)技術(shù)細節(jié)。建議大家拷貝文檔中的腳本實際運行一下,學習的

    2024年02月12日
    瀏覽(30)
  • Spring Boot異步任務(wù)、異步消息

    Spring Boot異步任務(wù)、異步消息

    目錄 1.異步任務(wù) 1.1.概述 1.2.使用 2.異步消息 2.1.概述 2.2.使用 舉一個例子,我現(xiàn)在有一個網(wǎng)上商城,客戶在界面點擊下單后,后臺需要完成兩步: 1.創(chuàng)建客戶訂單 2.發(fā)短信通知客戶訂單號 這里面第2步是個高耗時的操作,如果全部擠在一條主線程里做,效果就會是客戶點了一

    2023年04月22日
    瀏覽(33)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包