咦!Spring容器里为什么没有我需要的Bean?

咦!Spring容器里为什么没有我需要的Bean?

小故事

有一天,项目经理收到一个紧急需求,需要新增一个模块,项目经理看了看开发区的同学,一眼就看到盯着屏幕笑嘻嘻的小菜同学

项目经理心想:这傻小子在乐什么呢,肯定是在摸鱼,就让新需求给他做吧

项目经理悄咪咪的偷摸到小菜身后,看着小菜在沸点评论区不停的滑动,似乎在寻找着什么大瓜

此时的小菜似乎察觉到气氛不太对劲,身后似乎有人,于是飞快的按下 Windows + 1 弹出Idea的开发界面

此时,项目经理开口道:小菜啊,这里有个紧急需求,新增了一个模块,你新建一个SpringBoot项目把这些需求写一下吧~

小菜只能不情愿的接到需求,从而放弃摸鱼,开始开发

精通CRUD的小菜立马Copy出一个新项目

image.png

很快啊,啪的一下~写出启动类加上@SpringBootApplication注解

package com.caicaijava.springbooteasyframeworks;

@SpringBootApplication
public class SpringBootEasyFrameworksApplication {

	public static void main(String[] args) {
		SpringApplication.run(SpringBootEasyFrameworksApplication.class, args);
	}

}

然后小菜建立了一个controller包开始认真开发…

没过多久,小菜就把需求都搞定了,于是启动服务开始测试

小菜打开测试工具就开始测试接口,但是怎么测试都是404,一开始小菜还以为url写错了,但是检测后发现并没有写错

经过小菜漫长的排查,终于发现了问题:原来是小菜把controller包不小心创建在springbooteasyframeworks包的同级目录

image.png

启动类中的@SpringBootApplication中包含@ComponentScan

@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
		@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })

@ComponentScan会去扫描包下的组件,并把组件加入到容器中,由于没有配置basePackages字段,于是只会扫描当前包下的组件

当前包也就是com.caicaijava.springbooteasyframeworks

于是controller包与其同级时无法扫描其中的组件,因此导致容器中找不到对应的Bean

如果需要扫描其他包,或者需要依赖公共项目common下的包时,可以使用配置basePackages如果已经配置basePackages则不会默认扫描当前包,于是也要把当前包加上

@SpringBootApplication
@ComponentScan(basePackages = {"com.caicaijava.common","com.caicaijava.springbooteasyframeworks"})
public class SpringBootEasyFrameworksApplication {

    public static void main(String[] args) {
       SpringApplication.run(SpringBootEasyFrameworksApplication.class, args);
    }

}

总结

@SpringBootApplication默认只扫描当前包下的组件,如果需要扫描其他包,需要配置@ComponentScanbasePackagesvalue字段

当配置过@ComponentScan时,默认不会扫描当前包下的组件,注意把当前包也加入配置

🌠最后(一键三连求求拉~)

本篇文章被收入专栏 深入浅出常用框架,感兴趣的同学可以持续关注喔

本篇文章笔记以及案例被收入 Gitee-CaiCaiJavaGithub-CaiCaiJava,除此之外还有更多Java进阶相关知识,感兴趣的同学可以starred持续关注喔~

有什么问题可以在评论区交流,如果觉得菜菜写的不错,可以点赞、关注、收藏支持一下~

关注菜菜,分享更多技术干货,公众号:菜菜的后端私房菜

<think>好的,我现在需要帮助用户解决在Spring拦截器中如何获取容器中的Bean的问题。用户之前已经询问过过滤器和拦截器的区别,现在根据提供的引用内容,用户可能遇到了在拦截器中自动注入Bean为null的情况,想知道如何正确获取Bean。 首先,我要回顾用户提供的引用内容。引用[2]提到在拦截器中@Autowired注入的Bean为null,这可能是因为拦截器的实例化方式导致的。拦截器通常由Spring MVC配置,但如果没有正确注册到Spring容器中,拦截器本身不会被Spring管理,因此自动注入会失败。 接下来,我需要考虑解决方案。常见的解决方法包括: 1. **使用WebMvcConfigurer注册拦截器并手动注入Bean**:通过配置类将拦截器实例添加到Spring容器,并确保在注册拦截器时使用已经由Spring管理的Bean。例如,在配置类中使用@Bean创建拦截器实例,这样Spring会自动处理依赖注入。 2. **让拦截器实现BeanFactoryAware接口**:通过实现BeanFactoryAware接口,拦截器可以获取BeanFactory,从而通过它来获取需要的Bean。这种方法允许在拦截器内部手动获取Bean实例。 3. **使用ApplicationContextAware工具类**:创建一个工具类实现ApplicationContextAware,保存ApplicationContext的静态引用。然后在拦截器中通过这个工具类获取Bean。这种方法适用于需要全局访问ApplicationContext的情况。 此外,需要解释为什么直接使用@Autowired会失败,主要是因为拦截器可能没有Spring容器管理,导致依赖注入无法进行。需要强调拦截器必须由Spring创建实例,才能使用自动注入。 最后,要确保回答结构清晰,分步骤说明每种方法的实现,并给出代码示例。同时,提醒用户注意拦截器的注册方式,确保它们被Spring正确管理。</think>### Spring 拦截器中获取容器 Bean 的 3 种方法 在拦截器中直接使用 `@Autowired` 注解注入 Bean 会失败(如引用[2]所述),因为 **拦截器未被 Spring 容器管理**。以下是可用的解决方案: --- #### 方法 1:通过 `WebMvcConfigurer` 手动注入(推荐) 通过 Spring 配置类显式注册拦截器并注入依赖,确保拦截器实例由 Spring 管理。 ```java @Configuration public class WebConfig implements WebMvcConfigurer { @Autowired // 注入 Spring 容器中的 Bean private ToolsService toolsService; @Override public void addInterceptors(InterceptorRegistry registry) { // 手动创建拦截器实例并注入依赖 registry.addInterceptor(new MyInterceptor(toolsService)) .addPathPatterns("/api/**"); } } // 拦截器改造为通过构造函数接收依赖 public class MyInterceptor implements HandlerInterceptor { private final ToolsService toolsService; public MyInterceptor(ToolsService toolsService) { this.toolsService = toolsService; } @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { System.out.println(toolsService); // 正常输出 Bean 对象 return true; } } ``` **原理**:通过构造函数显式传递依赖,确保拦截器实例持有 Spring 管理的 Bean[^2]。 --- #### 方法 2:实现 `BeanFactoryAware` 接口 通过 Spring 的 `BeanFactoryAware` 接口直接获取容器对象。 ```java public class MyInterceptor implements HandlerInterceptor, BeanFactoryAware { private BeanFactory beanFactory; private ToolsService toolsService; @Override public void setBeanFactory(BeanFactory beanFactory) throws BeansException { this.beanFactory = beanFactory; // 延迟加载 Bean(避免初始化顺序问题) this.toolsService = beanFactory.getBean(ToolsService.class); } @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { System.out.println(toolsService); // 正常输出 return true; } } ``` **注意**:需在 Spring 配置类中注册该拦截器,确保 Spring 调用 `setBeanFactory` 方法[^2]。 --- #### 方法 3:使用 `ApplicationContextAware` 工具类 创建全局工具类获取 Spring 上下文,通过上下文获取 Bean。 ```java @Component public class SpringContextUtil implements ApplicationContextAware { private static ApplicationContext applicationContext; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { SpringContextUtil.applicationContext = applicationContext; } public static <T> T getBean(Class<T> clazz) { return applicationContext.getBean(clazz); } } // 拦截器中调用工具类 public class MyInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { ToolsService toolsService = SpringContextUtil.getBean(ToolsService.class); System.out.println(toolsService); // 正常输出 return true; } } ``` **适用场景**:需要跨多个非 Spring 管理类获取 Bean 的情况。 --- ### 核心原理 - **拦截器未被 Spring 管理**:默认情况下,拦截器实例由 MVC 框架直接创建,而非通过 Spring 容器,因此无法自动注入。 - **依赖传递**:通过构造函数或工具类显式传递 Spring 容器中的 Bean,绕过自动注入的限制。 --- ### 扩展问题 1. 为什么 `@Component` 注解无法让拦截器自动注入 Bean? 2. 如何避免 `ApplicationContextAware` 工具类引起的循环依赖? 3. 拦截器中使用 `@Autowired` 注解的 Bean 何时会初始化?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值