当前位置: 首页 > news >正文

Spring拦截器中@Resource注入为null的问题

Spring拦截器中@Resource注入为null的问题

前言

最近在跟着黑马的视频学SpringBoot项目,遇到了一个很奇怪的问题。在写登录拦截器的时候,明明用了@Resource注解,但是StringRedisTemplate一直是null,调试了好久才找到原因。记录一下这个坑,希望能帮到同样遇到问题的你们。

问题描述

我按照平时的写法,想在拦截器里注入Redis模板:

@Component
public class LoginInterceptor implements HandlerInterceptor {@Resourceprivate StringRedisTemplate stringRedisTemplate; // 这里一直是null!@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 运行到这里就报空指针异常stringRedisTemplate.opsForHash().entries("test");return true;}
}

配置类是这样写的:

@Configuration
public class MvcConfig implements WebMvcConfigurer {@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new LoginInterceptor()); // 问题就在这里!}
}

为什么会这样?

刚开始我以为是Redis配置有问题,因为在Service层用同样的注解是可以的。问题出在对Spring工作原理的理解上。

Spring依赖注入的条件

Spring只会对它自己管理的Bean进行依赖注入。什么意思呢?

  • 当我写@Component的时候,Spring确实会创建一个LoginInterceptor的Bean
  • 但是在MvcConfig里,我用的是new LoginInterceptor(),这是手动创建的对象
  • 手动创建的对象和Spring管理的Bean是两个不同的实例
  • Spring只会对容器里的那个Bean注入依赖,对手动创建的对象不管

所以就出现了这种情况:

// Spring容器里有一个LoginInterceptor,StringRedisTemplate被正确注入
LoginInterceptor springBean = 容器中的实例; // stringRedisTemplate ✓// 我手动创建了另一个LoginInterceptor,Spring不管它
LoginInterceptor myObject = new LoginInterceptor(); // stringRedisTemplate = null ✗

解决方法

方法一:构造函数传参(推荐的方法)

public class LoginInterceptor implements HandlerInterceptor {private StringRedisTemplate stringRedisTemplate;// 通过构造函数接收StringRedisTemplatepublic LoginInterceptor(StringRedisTemplate stringRedisTemplate) {this.stringRedisTemplate = stringRedisTemplate;}@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {// 现在可以正常使用了stringRedisTemplate.opsForHash().entries("test");return true;}
}

配置类改成这样:

@Configuration
public class MvcConfig implements WebMvcConfigurer {@Resourceprivate StringRedisTemplate stringRedisTemplate;@Overridepublic void addInterceptors(InterceptorRegistry registry) {// 把StringRedisTemplate传给拦截器registry.addInterceptor(new LoginInterceptor(stringRedisTemplate));}
}

方法二:注入拦截器Bean

@Component
public class LoginInterceptor implements HandlerInterceptor {@Resourceprivate StringRedisTemplate stringRedisTemplate;// 其他代码不变
}

配置类这样写:

@Configuration
public class MvcConfig implements WebMvcConfigurer {@Resource // 直接注入Spring管理的LoginInterceptorprivate LoginInterceptor loginInterceptor;@Overridepublic void addInterceptors(InterceptorRegistry registry) {// 用注入的实例,不要newregistry.addInterceptors(loginInterceptor);}
}

为什么选择方法一?

为什么不用方法二

  1. 更直观:一看就知道拦截器需要什么依赖
  2. 更灵活:不需要给拦截器加@Component
  3. 更好测试:写单元测试的时候容易mock
  4. 避免错误:不容易忘记加注解而导致注入失败

确实,构造函数的方式让依赖关系很清楚,代码也更容易理解。

学到的经验

  1. Spring只对自己管理的Bean注入依赖,手动new的对象不行
  2. 看到@Resource@Autowired为null,先检查对象是不是Spring管理的
  3. 拦截器、过滤器这些需要手动创建的组件,通过构造函数传依赖比较好
  4. 遇到问题先想想Spring的工作原理,很多问题都能迎刃而解

总结

如果你也遇到了类似的依赖注入问题,可以按这个思路检查:

  1. 这个对象是Spring管理的吗?
  2. 如果不是,怎么让Spring管理它?
  3. 或者换个方式传递依赖?
http://www.dtcms.com/a/349952.html

相关文章:

  • Github热门开源项目榜单 - 2025年07月
  • 【c++】leetcode300 最长递增子序列
  • 二、添加3D形状
  • Springboot应用如何与SkyWalking集成,并使用Docker进行发布
  • 深入理解Linux进程程序替换:从原理到实践
  • Elasticsearch JVM调优:核心参数与关键技巧
  • Git克隆时遇到“Filename too long“错误的完美解决方案
  • 代理设计模式
  • 俄罗斯情报机构推出新型安卓恶意软件,伪装成杀毒软件
  • SciPy科学计算与应用:SciPy入门与应用-科学计算与NumPy协同实践
  • 工业异常检测大模型(1)数据集、方法
  • 【git使用场景】本地仓库与远程仓库存在独立历史
  • Vulkan 学习路线图
  • Git 怎么仓库迁移?error: remote origin already exists.怎么解决
  • 定时器的原理
  • TensorFlow 深度学习 | Dataset API 数据读取详解
  • Open3D入门指南:3D数据处理与可视化利器
  • 初识神经网络——《深度学习入门:基于Python的理论与实现》
  • 昆仑万维开源 Matrix-3D大模型,正在开启“造物主”模式
  • 【智慧城市】2025年中国地质大学(武汉)暑期实训优秀作品(2):智慧城市西安与一带一路
  • pytest 并发执行用例(基于受限的测试资源)
  • imx6ull-驱动开发篇40——Linux RTC 驱动简介
  • 一道MySQL笔试题: 输出 100 以内质数
  • VIVO/OPPO手机,显示5G开关
  • 【SystemUI】锁屏来通知默认亮屏Wake模式
  • Mac 菜单栏多合一工具自荐:FancyTool
  • LeetCode算法日记 - Day 22: 提莫攻击、Z字形变换
  • 电影感人文街拍摆摊纪实摄影后期Lr调色教程,手机滤镜PS+Lightroom预设下载!
  • 从手术室到街头摄像头:多模态融合如何让AI“看得懂”万物?
  • 搭建ftp服务器(主动模式,被动模式)