@Transaction,@Async在同一個(gè)類中注解失效的原因分析及解決
@Transaction @Async在同一個(gè)類中注解失效
在同一個(gè)類中,一個(gè)方法調(diào)用另外一個(gè)有注解(比如@Async,@Transational)的方法,注解是不會(huì)生效的。
比如,下面代碼例子中,有兩方法,一個(gè)有@Async注解,一個(gè)沒有。第一次如果調(diào)用了有注解的test()方法,會(huì)啟動(dòng)@Async注解作用;第一次如果調(diào)用testAsync(),因?yàn)樗鼉?nèi)部調(diào)用了有注解的test(),如果你以為系統(tǒng)也會(huì)為它啟動(dòng)Async作用,那就錯(cuò)了,實(shí)際上是沒有的。
@Service
public class TestAsyncService {
public void testAsync() throws Exception {
test();
}
@Async
public void test() throws InterruptedException{
Thread.sleep(10000);//讓線程休眠,根據(jù)輸出結(jié)果判斷主線程和從線程是同步還是異步
System.out.println("異步threadId:"+Thread.currentThread().getId());
}
}
運(yùn)行結(jié)果:testAsync()主線程和從線程()test()從線程同步執(zhí)行。
原因:spring 在掃描bean的時(shí)候會(huì)掃描方法上是否包含@Async注解,如果包含,spring會(huì)為這個(gè)bean動(dòng)態(tài)地生成一個(gè)子類(即代理類,proxy),代理類是繼承原來那個(gè)bean的。
此時(shí),當(dāng)這個(gè)有注解的方法被調(diào)用的時(shí)候,實(shí)際上是由代理類來調(diào)用的,代理類在調(diào)用時(shí)增加異步作用。
然而,如果這個(gè)有注解的方法是被同一個(gè)類中的其他方法調(diào)用的,那么該方法的調(diào)用并沒有通過代理類,而是直接通過原來的那個(gè)bean,所以就沒有增加異步作用,我們看到的現(xiàn)象就是@Async注解無效。
下面用偽代碼闡述一下原因
@Service
class A{
@Async
method b(){...}
method a(){ //標(biāo)記1
b();
}
}
//Spring掃描注解后,創(chuàng)建了另外一個(gè)代理類,并為有注解的方法加上異步效果
class proxy$A{
A objectA = new A();
method b(){ //標(biāo)記2
//異步執(zhí)行Async
objectA.b();
}
method a(){ //標(biāo)記3
objectA.a(); //由于a()沒有注解,所以不會(huì)異步執(zhí)行,而是直接調(diào)用A的實(shí)例的a()方法
}
}
當(dāng)我們調(diào)用A的bean的a()方法的時(shí)候,也是被proxyA攔截,執(zhí)行proxyA攔截,執(zhí)行proxyA.a()(標(biāo)記3),然而,由以上代碼可知,這時(shí)候它調(diào)用的是objectA.a(),也就是由原來的bean來調(diào)用a()方法了,所以代碼跑到了“標(biāo)記1”。由此可見,“標(biāo)記2”并沒有被執(zhí)行到,所以異步執(zhí)行的效果也沒有運(yùn)行。
說說解決
了解了失效的原因,解決的方法就簡單了(兩種):
- 把這兩個(gè)方法分開到不同的類中
- 把注解加到類名上面
@Async的實(shí)現(xiàn)類方式
詳解:用于開啟異步處理的接口, @Async,使用異步必須再啟動(dòng)類加上@EnableAsync
方法1:實(shí)現(xiàn)接口AsyncConfigurer
@Configuration
public class ThreadConfiguration implements AsyncConfigurer {
private Logger logger = LoggerFactory.getLogger(this.getClass());
private final ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
@Bean
public Executor getAsyncExecutor() {
executor.setThreadNamePrefix("Zy-TaskExecutor-");
executor.setCorePoolSize(50);// 核心線程池大小
executor.setMaxPoolSize(200);// 最大可創(chuàng)建的線程數(shù)
executor.setQueueCapacity(1000);// 隊(duì)列最大長度
executor.setKeepAliveSeconds(300);// 線程池維護(hù)線程所允許的空閑時(shí)間
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
executor.initialize();
logger.info("任務(wù)線程池初始化...");
return executor;
}
/**
* 只有在方法上添加@Async的出現(xiàn)異常才會(huì)跳到此方法中
* */
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
logger.error("線程執(zhí)行出現(xiàn)異常....");
return (e, method, arguments) -> logger.error("exception method : " + method.getName() + " message:" + e.getMessage(), e);
}
/**
* 線程監(jiān)控類,訪問/monitor
* */
@Bean
public ServletRegistrationBean threadPoolMonitorServlet() {
ServletRegistrationBean registration = new ServletRegistrationBean(new HttpServlet() {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType(MediaType.APPLICATION_JSON_UTF8_VALUE);
PrintWriter writer = resp.getWriter();
writer.print("corePoolSize : " + executor.getCorePoolSize());
writer.flush();
writer.close();
}
});
registration.addUrlMappings("/monitor/*");
logger.info("springboot線程監(jiān)控start!");
return registration;
}
}
方法2:直接注入bean
@Bean
public ThreadPoolTaskExecutor threadPoolTaskExecutor(){
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setMaxPoolSize(200);
executor.setQueueCapacity(1000);
executor.setCorePoolSize(50);
executor.setKeepAliveSeconds(300);
executor.setThreadNamePrefix("Zy-TaskExecutor-");
// 線程池對(duì)拒絕任務(wù)(無線程可用)的處理策略
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
executor.initialize();
log.info("任務(wù)線程池初始化...");
return executor;
}
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java中pdf轉(zhuǎn)圖片的實(shí)現(xiàn)方法
下面小編就為大家?guī)硪黄猨ava中pdf轉(zhuǎn)圖片的實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-12-12
Mybatis 中Mapper使用package方式配置報(bào)錯(cuò)的解決方案
這篇文章主要介紹了Mybatis 中Mapper使用package方式配置報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07
Java 批量獲取地址間距離工具(支持中轉(zhuǎn)站)
本文主要介紹了Java批量獲取地址間距離,獲取兩個(gè)地址間距離,實(shí)現(xiàn)方式比較多,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07
詳解如何在低版本的Spring中快速實(shí)現(xiàn)類似自動(dòng)配置的功能
這篇文章主要介紹了詳解如何在低版本的Spring中快速實(shí)現(xiàn)類似自動(dòng)配置的功能,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-05-05
Spring boot如何集成kaptcha并生成驗(yàn)證碼
這篇文章主要介紹了Spring boot如何集成kaptcha并生成驗(yàn)證碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07
Java中檢查值是否存在于數(shù)組中的4種詳細(xì)方法
這篇文章主要給大家介紹了關(guān)于Java中檢查值是否存在于數(shù)組中的4種詳細(xì)方法,相信大家在操作Java的時(shí)候經(jīng)常會(huì)要檢查一個(gè)數(shù)組(無序)是否包含一個(gè)特定的值,需要的朋友可以參考下2023-08-08

