springboot拦截器HandlerInterceptor的注入问题

一、问题描述

   项目启动后,拦截器下的注入为null,影响拦截器的逻辑操作。如下图:

二、解决策略

   原因: 拦截器加载是在springcontext创建之前完成,详情可以看spring的拦截器加载过程及IOC的关系

  解决方案一:使用@Bean在拦截器初始化之前让类加载,重点在于@Bean的拦截器处理及getCheckFilter()的引入

 具体如下:

// 拦截器处理
public class CheckFilter implements HandlerInterceptor {

    @Autowired
    private SystemService systemService;
}


// 配置类处理

@Configuration
public class WebMvcConfig extends WebMvcConfigurationSupport {

    @Bean
    public CheckFilter getCheckFilter(){
        return  new CheckFilter();
    }

    @Override
    protected void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(getCheckFilter()).addPathPatterns("/**").
                excludePathPatterns("/admin/auth/**");
        super.addInterceptors(registry);
    }

    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("/static/**").addResourceLocations("classpath:/static/");
        super.addResourceHandlers(registry);
    }

    @Override
    protected void configureViewResolvers(ViewResolverRegistry registry) {
        registry.jsp("/WEB-INF/",".jsp");
        super.configureViewResolvers(registry);
    }
}

解决方案二:给拦截器增加配置注解如:@Configuration  、@Component 。在配置类中注入,具体如下操作:

// 拦截器处理
@Configuration   //或者 @Component
public class CheckFilter implements HandlerInterceptor {

    @Autowired
    private SystemService systemService;
}


// 配置类处理

@Configuration
public class WebMvcConfig extends WebMvcConfigurationSupport {

    @Autowired
    private CheckFilter  checkFilter;

    @Override
    protected void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(checkFilter).addPathPatterns("/**").
                excludePathPatterns("/admin/auth/**");
        super.addInterceptors(registry);
    }

    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
 registry.addResourceHandler("/static/**").addResourceLocations("classpath:/static/");
        super.addResourceHandlers(registry);
    }

    @Override
    protected void configureViewResolvers(ViewResolverRegistry registry) {
        registry.jsp("/WEB-INF/",".jsp");
        super.configureViewResolvers(registry);
    }
}

三、知识总结

  1、涉及spring的拦截器加载过程及IOC的管理

  2、参考资料https://blog.csdn.net/dengdeng333/article/details/87878882

### 如何解决 Spring 拦截器中 Bean 注入失败问题 当遇到在拦截器中无法注入 `Bean` 的情况时,主要原因是由于拦截器被手动实例化而不是由 Spring 容器管理。这导致了即使使用了 `@Component` 或者其他注解也无法正常注入所需的 `Bean`。 #### 方法一:让 Spring 管理拦截器的生命周期 为了使拦截器能够成功获取到容器中的 `Bean`,应该避免直接通过 `new` 关键字创建拦截器对象。相反,在定义拦截器配置类时应当利用 `addInterceptors()` 方法来注册拦截器,并确保该方法返回的是由 Spring 创建并托管的对象而非自行构建的新实例[^1]。 ```java @Configuration public class WebConfig implements WebMvcConfigurer { private final MyInterceptor myInterceptor; public WebConfig(MyInterceptor myInterceptor){ this.myInterceptor = myInterceptor; } @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(myInterceptor); } } ``` 这样做的好处在于可以让 Spring 自动完成依赖关系的装配工作,从而使得所有的 `Bean` 都可以按照预期的方式运作起来。 #### 方法二:调整初始化顺序 另一种解决方案是改变应用程序启动过程中某些组件加载的时间点。具体来说就是在应用上下文刷新之后再向框架内添加自定义逻辑(比如设置拦截器)。可以通过监听事件的方式来实现这一点: ```java @Component public class ApplicationStartupListener implements ApplicationListener<ApplicationReadyEvent> { private final InterceptorRegistry interceptorRegistry; public ApplicationStartupListener(InterceptorRegistry interceptorRegistry){ this.interceptorRegistry = interceptorRegistry; } @Override public void onApplicationEvent(ApplicationReadyEvent event) { // 此处可安全地操作interceptorRegistry,因为此时所有beans都已准备好 interceptorRegistry.addInterceptor(new CustomInterceptor()); } } ``` 不过这种方法相对复杂些,通常推荐采用第一种方式即保证拦截器是由Spring负责管理和实例化的[^2]。 #### 方法三:延迟初始化 对于那些确实需要提前注册但又不想影响整体流程的情况,则可以选择延迟初始化策略。这意味着只有当实际请求到来时才会真正去尝试访问相应的资源和服务。这种方式适用于特定场景下的优化需求[^4]。 ```java @RestControllerAdvice public class GlobalExceptionHandler extends ResponseEntityExceptionHandler { @Autowired(required=false) private transient RedisTemplate<String, Object> redisTemplate; @PostConstruct protected void init() { if (redisTemplate == null) { ApplicationContext context = ... ;// 获取ApplicationContext的方法 redisTemplate = context.getBean(RedisTemplate.class); } } } ``` 以上三种途径都可以有效地应对 Spring Boot 中常见的拦截器 `Bean` 注入失败的问题。选择哪种取决于具体的业务背景和个人偏好。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值