"最近在使用 springboot 拦截器的过程当中,发现无法注入 service, 提示 NPE 错误 问题相关代码 springboot 拦截器注册方式: @Configuration public class MyWebMvcConfigurer implements WebMvcConfigurer { @Ove .."

springboot 拦截器自动注入为 null 解决办法和具体原因分析

最近在使用 springboot 拦截器的过程当中,发现无法注入 service, 提示 NPE 错误

问题相关代码

springboot 拦截器注册方式:

@Configuration
public class MyWebMvcConfigurer implements WebMvcConfigurer {
	
  @Override
  public void addInterceptors(InterceptorRegistry registry) {
    registry.addInterceptor(new SessionInterceptor()).excludePathPatterns("/static/**").addPathPatterns("/**");
  }
}

拦截器代码:

public class SessionInterceptor  implements HandlerInterceptor {
    @Autowired
    private IUserService  userService;

    @Override
    public boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o) throws Exception {
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {
        HttpSession session = httpServletRequest.getSession();
        String username = (String)session.getAttribute("userName");
        //获取请求信息的用户名获取对应的用户信息
        SysUser userInfo =userService.getUserInfoByUserName(username).get(0);
    }

项目运行,发起请求,进入拦截器,直接报错 NPE,userService 为 null。

解决方式

方式一:使用 WebApplicationContext 上下文对象来手动注入
方式二:通过 @bean 注解注入拦截器到 spring 当中, 同时给拦截器加上 @compent 注解

@Configuration
public class MyWebMvcConfigurer implements WebMvcConfigurer {
  @Bean
  public SessionInterceptor getSessionInterceptor(){
    return new SessionInterceptor();
  }

  @Override
  public void addInterceptors(InterceptorRegistry registry) {
    registry.addInterceptor(getSessionInterceptor()).excludePathPatterns("/static/**").addPathPatterns("/**");
  }
}

问题原因(划重点)

在网上找了很多资料,看到不少文章说注入为 null 的原因是:

造成 null 的原因是因为拦截器加载是在 springcontext 创建之前完成的,所以在拦截器中注入实体自然就为 null。

当看到这个答案的时候,我是有疑惑的,因为在拦截器当中是可以注入 bean 的。带着疑惑,找了很多资料:比如 springboot 的加载顺序,拦截器等等,也问了很多人,最终得到了一个我个人比较认可的答案:

注入为 null 的时候,是通过 new 的方式创建的拦截器,通过 new 出来的实例是没有交给 spring 进行管理的,没有被 spring 管理的实例,spring 是无法自动注入 bean 的,所以为 null

  • Spring

    Spring 是一个开源框架,是于 2003 年兴起的一个轻量级的 Java 开发框架,由 Rod Johnson 在其著作《Expert One-On-One J2EE Development and Design》中阐述的部分理念和原型衍生而来。它是为了解决企业应用开发的复杂性而创建的。框架的主要优势之一就是其分层架构,分层架构允许使用者选择使用哪一个组件,同时为 JavaEE 应用程序开发提供集成的框架。

    551 引用 • 1168 回帖 • 886 关注
  • 拦截器
    2 引用 • 12 回帖
回帖   
请输入回帖内容...