Spring Boot中@Autowired注入不进去的问题

本文介绍了解决SpringBoot应用中多线程环境下@Autowired注解无法正确注入Bean的问题。通过创建一个工具类ApplicationContextProvider,实现了在多线程环境中获取所需Bean的功能,确保了线程安全。

使用Spring Boot 用@Autowired注入时,报空指针异常,经过排查,发现自己的DAO并没有注入成功,打印Dao接口为NULL。后来发现,在线程中为了线程安全,是防注入的。多线程时,注入是不成功的。然后写了一个工具类来获取Bean。代码如下:

package com.chen.util;

import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

@Component
public class ApplicationContextProvider implements ApplicationContextAware {
    /**
     * 上下文对象实例
     */
    private static ApplicationContext applicationContext;
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
    /**
     * 获取applicationContext
     *
     * @return
     */
    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }
    /**
     * 通过name获取 Bean.
     *
     * @param name
     * @return
     */
    public static Object getBean(String name) {
        return getApplicationContext().getBean(name);
    }
    /**
     * 通过class获取Bean.
     *
     * @param clazz
     * @param <T>
     * @return
     */
    public static <T> T getBean(Class<T> clazz) {
        return getApplicationContext().getBean(clazz);
    }
    /**
     * 通过name,以及Clazz返回指定的Bean
     *
     * @param name
     * @param clazz
     * @param <T>
     * @return
     */
    public static <T> T getBean(String name, Class<T> clazz) {
        return getApplicationContext().getBean(name, clazz);
    }
}

通过这个工具类可以正常获取到想要的Bean,new的地方也可以用

public test() {
  //new的时候注入需要的bean
  AdminDao adminDao= ApplicationContextProvider.getBean(AdminDao.class);
 }
<think>好的,我现在需要解决用户在Spring Boot中遇到的@Autowired注入失败的问题。用户已经提供了一些相关的引用资料,我需要结合这些信息来整理可能的原因和解决方案。 首先,我应该回顾一下常见的@Autowired注入失败的原因。根据引用[1],可能的原因包括没有被Spring管理、包扫描范围正确、循环依赖、静态变量注入等。引用[2]提到包扫描的问题,特别是当在启动的子包下时,需要用scanBasePackages手动指定。引用[3]和[4]则讨论了静态变量注入问题以及拦截器中需要手动配置Bean的情况。 接下来,我需要将这些原因分门别,并给出对应的解决方案。例如,包扫描问题需要检查启动的扫描配置;静态变量能直接使用@Autowired,需要通过setter方法或@PostConstruct来初始化。另外,还要考虑组件是否被正确注解,比如是否遗漏了@Service、@Component等。 然后,用户可能还遇到其他问题,比如多实例情况下的注入冲突,或者使用new关键字创建对象导致Spring无法管理。这些也需要涵盖在回答中。 需要注意的是,解决方案要具体,例如在包扫描的问题中,如何修改启动注解,或者如何配置拦截器的Bean。同时,要提醒用户检查依赖是否完整,比如是否缺少必要的starter依赖。 最后,总结常见原因和解决步骤,帮助用户系统地排查问题。确保回答结构清晰,逐步引导用户检查和解决问题,同时引用提供的资料来支持每个解决方案的正确性。</think>### Spring Boot@Autowired注入失败原因及解决方案 #### 一、常见原因分析 1. **未注册Spring组件** - 被注入未添加$@Component$、$@Service$、$@Repository$或$@Controller$等注解,导致未被Spring管理[^1]。 - 示例:未标注$@Service$的服务无法被注入。 2. **包扫描范围足** - Spring Boot默认扫描启动所在包及其子包。若目标在该范围内,需通过$scanBasePackages$显式指定扫描路径[^2]。 - 修改启动: ```java @SpringBootApplication(scanBasePackages = {"com.example.core", "com.example.web"}) ``` 3. **静态变量直接注入** - $@Autowired$无法直接注入静态变量。需通过非静态方法间接赋值[^3]。 - 示例代码: ```java @Service public class MyService { private static MyComponent myComponent; @Autowired public void setMyComponent(MyComponent component) { MyService.myComponent = component; } } ``` 4. **拦截器/过滤器等特殊场景** - 拦截器通过$new$实例化时,Spring会自动注入依赖。需通过$@Bean$手动注册[^4]。 - 配置示例: ```java @Configuration public class WebConfig implements WebMvcConfigurer { @Bean public MyInterceptor myInterceptor() { return new MyInterceptor(); } @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(myInterceptor()); } } ``` 5. **循环依赖或Bean冲突** - 多个同型Bean未指定$@Primary$或$@Qualifier$时,导致注入歧义。 - 解决方案:明确指定Bean名称或优先级。 --- #### 二、系统化排查步骤 1. **检查注解** - 确认被注入注入目标均添加了Spring管理注解(如$@Service$)。 2. **验证包扫描范围** - 若位于启动同级或子包外,需扩展$scanBasePackages$[^2]。 3. **避免直接操作静态变量** - 改用实例方法+静态变量赋值方式[^3]。 4. **特殊组件手动注册** - 拦截器、过滤器等需通过$@Bean$声明[^4]。 5. **依赖完整性检查** - 确认$pom.xml$中是否包含相关Spring Boot Starter依赖(如$spring-boot-starter-web$)。 --- #### 三、典型场景示例 **场景:WebSocket中注入Service失败** ```java @Component public class WebSocketHandler { private static MyService myService; @Autowired public void setMyService(MyService service) { myService = service; // 通过实例方法注入静态变量 } } ``` **场景:多数据源配置冲突** ```java @Primary // 标记主数据源 @Bean public DataSource primaryDataSource() { return DataSourceBuilder.create().build(); } ``` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值