ApplicationContext获取不到Bean的问题

文章讨论了在SpringBoot1.x环境下遇到的NoSuchBeanDefinitionException异常,问题源于在内部调用带@Transactional注解的方法时,使用了applicationContext.getBean导致的问题。解决方案包括升级SpringBoot到2.x(默认使用CGLIB代理)或手动指定使用CGLIB,以及通过修改代码结构避免直接调用bean。通过AOP切面验证了事务注解的效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

开发环境遇到报错:
org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'XXXXX' available

现将原问题代码简化抽出分析。

部署环境:

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)从而使用了动态代理,该环境使用的是JDK的动态代理机制,那么代理不会继承而是使用实现接口。所以这个bean是通过接口注入的,就无法通过接口的实例与bean的实例关联。
ps:

  1. Springboot 1.x AOP默认还是使用 JDK 动态代理的
  2. 如果目标对象实现了接口默认采用JDK动态代理 (可强制改为CGLIB)
  3. 如果目标对象没有实现接口默认采用CGLIB动态代理
解决方案:
  1. 升级SpringBoot版本为2.x,AOP默认使用CGLIB实现。
  2. properties配置文件增加:spring.aop.proxy-target-class=true ## 强制使用CGLIB代理
  3. 使用applicationContext.getBean的重载方法获取对应的bean
修复代码

为了探求transactional注解是否生效,现增加AOP 切面类模拟事物。

@Aspect
@Configuration
public class AopConfig {

    /**
     * 增强transactionalMethod方法 模拟transational注解
     */
    @Pointcut("execution(* com.example.springbootdemo.error.test.ErrorBeanServiceImpl.transactionalMethod())")
    public void executeAdvice() {
    }

    /**
     * 环绕增强
     */
    @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();
    }
}
  1. 升级SpringBoot版本为2.x 略
  2. 配置配置文件 略
  3. 修改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

体悟:八股文还是有用的~

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值