解決ApplicationContext獲取不到Bean的問題
開發(fā)環(huán)境遇到報錯
org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'XXXXX' available
現將原問題代碼簡化抽出分析。
部署環(huán)境
- springboot版本:1.x
- java:1.8
- 其他:略
問題代碼
service層
- 接口:
public interface ErrorBeanService { void transactionalMethod(); void callMethod(); }
- 實現類:
@Service("errorBeanServiceImpl") public class ErrorBeanServiceImpl implements ErrorBeanService{ private ApplicationContext applicationContext; @Autowired public void setApplicationContext(ApplicationContext applicationContext){ this.applicationContext = applicationContext; } @Transactional(rollbackFor = Exception.class) public void transactionalMethod() { System.out.println("transactionalMethod run ~ "); } public void callMethod() { // Transactional注解失效 // transactionalMethod(); // 使用applicationContext獲取到bean 使事物注解生效 applicationContext.getBean(ErrorBeanServiceImpl.class).transactionalMethod(); // 報錯行 } }
contro層:
@RestController @RequestMapping("/error") public class ErrorController { @Autowired private ErrorBeanService errorBeanService; @RequestMapping(value = "/errorBean", method = RequestMethod.GET) public void errorBean() { errorBeanService.callMethod(); } }
查看報錯行代碼,使用了applicationContext.getBean的方式獲取bean使得內部調用事務注解生效,但在獲取bean時報錯NoSuchBeanDefinitionException異常,報錯如下:
排除了包掃描和beanName問題,看上述報錯倒數第四行報錯:
org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:215)
由于使用了事物注解(AOP)從而使用了動態(tài)代理,該環(huán)境使用的是JDK的動態(tài)代理機制,那么代理不會繼承而是使用實現接口。
所以這個bean是通過接口注入的,就無法通過接口的實例與bean的實例關聯。
ps:
- Springboot 1.x AOP默認還是使用 JDK 動態(tài)代理的
- 如果目標對象實現了接口默認采用JDK動態(tài)代理 (可強制改為CGLIB)
- 如果目標對象沒有實現接口默認采用CGLIB動態(tài)代理
解決方案
- 升級SpringBoot版本為2.x,AOP默認使用CGLIB實現。
- properties配置文件增加:spring.aop.proxy-target-class=true ## 強制使用CGLIB代理
- 使用applicationContext.getBean的重載方法獲取對應的bean
修復代碼
為了探求transactional注解是否生效,現增加AOP 切面類模擬事物。
@Aspect @Configuration public class AopConfig { /** * 增強transactionalMethod方法 模擬transational注解 */ @Pointcut("execution(* com.example.springbootdemo.error.test.ErrorBeanServiceImpl.transactionalMethod())") public void executeAdvice() { } /** * 環(huán)繞增強 */ @Around("executeAdvice()") public Object aroundAdvice(ProceedingJoinPoint thisJoinPoint) throws Throwable { System.out.println("aroundAdvice before proceed"); Object obj = thisJoinPoint.proceed(); System.out.println("aroundAdvice after proceed"); return obj; } }
去除事務注解:
@Service("errorBeanServiceImpl") public class ErrorBeanServiceImpl implements ErrorBeanService{ private ApplicationContext applicationContext; @Autowired public void setApplicationContext(ApplicationContext applicationContext){ this.applicationContext = applicationContext; } public void transactionalMethod() { System.out.println("transactionalMethod run ~ "); } public void callMethod() { applicationContext.getBean(ErrorBeanServiceImpl.class).transactionalMethod(); } }
- 升級SpringBoot版本為2.x 略
- 配置配置文件 略
- 修改callMethod方法使用getBean的重載方法
<T> T getBean(String var1, Class<T> var2) throws BeansException;
代碼如下:
public void callMethod() { // applicationContext.getBean(ErrorBeanServiceImpl.class).transactionalMethod(); applicationContext.getBean("errorBeanServiceImpl", ErrorBeanService.class).transactionalMethod(); }
上述三種方式最終輸出結果:
aroundAdvice before proceed
transactionalMethod run ~
aroundAdvice after proceed
體悟:八股文還是有用的~
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Spring-基于Spring使用自定義注解及Aspect實現數據庫切換操作
這篇文章主要介紹了Spring-基于Spring使用自定義注解及Aspect實現數據庫切換操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09使用Spring AntPathMatcher的doMatch方法
這篇文章主要介紹了使用Spring AntPathMatcher的doMatch方法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09