目录
- 一、背景
- 二、排查过程及原因
- (一)、排查过程
- (二)、原因
- 三、解决方法
- 四、总结
一、背景
项目没有改任何配置文件,突然测试环境启动不起来,报一堆和redis相关的错误,但是本地启动没有任何问题。都是相同的代码。
二、排查过程及原因
(一)、排查过程
1、一开始怀疑是数据库或者redis相关配置文件被人为修改了,经过检查发现没有变动
2、通过redis报错的相关代码加入日志,发现注入的RedisTemplate为null
查看服务器日志
检查是否有以下错误:
biRedisTemplate Bean 初始化失败(如配置错误)。
调用 BIRedisUtil 静态方法时的空指针异常堆栈。
添加日志跟踪
在 setRedisTemplate() 方法中记录注入是否成功:
@Autowired
@Qualifier("biRedisTemplate")
public void setRedisTemplate(RedisTemplate redisTemplate) {log.info("Injecting biRedisTemplate: {}", redisTemplate);BIRedisUtil.redisTemplate = redisTemplate;
}
若服务器日志中无此信息,说明注入未触发。
静态方法添加空指针保护
public static String get(String key) {if (stringRedisTemplate == null) {log.error("stringRedisTemplate is null!");return null;}return stringRedisTemplate.opsForValue().get(key);
}
(二)、原因
- Spring 依赖注入机制不直接支持静态变量
Spring 的 @Autowired 注解默认针对实例成员,而非静态变量。
虽然文件2通过 setter 方法尝试将 Bean 注入静态变量:
@Autowired
@Qualifier("biRedisTemplate")
public void setRedisTemplate(RedisTemplate redisTemplate) {BIRedisUtil.redisTemplate = redisTemplate; // 赋值给静态变量
}
但以下问题会导致注入失败:
- 初始化顺序不可控
Spring 容器初始化时,静态变量的赋值可能晚于其他 Bean 的初始化。
例如:若某个 Bean 在初始化时直接调用 BIRedisUtil.batchQuery(),而 setRedisTemplate() 方法尚未执行,此时 redisTemplate 仍为 null。
静态变量的生命周期与 Bean 不同步 静态变量在类加载时被初始化为 null,而 Spring 依赖注入发生在 Bean 实例化后。若注入过程中发生异常(如 Bean 未定义或配置错误),静态变量会保持 null。
- 本地与服务器环境的差异
本地环境
可能因以下原因掩盖问题:
测试数据量小,未触发多线程并发场景。 依赖的 Bean 初始化顺序恰好正确(如先调用 setRedisTemplate() 后使用静态方法)。 Redis 配置简单,Bean 初始化无异常。
服务器环境
以下问题会暴露缺陷:
复杂依赖链:Bean 初始化顺序更复杂,可能先使用静态方法后注入变量。 并发请求:高并发下多个线程同时访问未初始化的redisTemplate,导致空指针。 配置错误:服务器 Redis 连接参数(如地址、密码)错误,导致 biRedisTemplate> Bean 初始化失败,setRedisTemplate() 未被调用。
- 静态变量的线程安全问题
即使注入成功,RedisTemplate 作为静态变量被多线程共享时: 若 RedisTemplate内部状态非线程安全(如连接池管理),可能导致数据错乱。 服务器高并发场景下,问题会被放大,而本地低并发测试可能无法复现。
三、解决方法
- 改为实例变量(推荐)
移除 static 关键字,通过 Spring 管理实例:
@Component
public class BIRedisUtil {private RedisTemplate redisTemplate; // 非静态变量@Autowiredpublic void setRedisTemplate(RedisTemplate redisTemplate) {this.redisTemplate = redisTemplate; // 注入实例变量}
}
- 强制初始化顺序(不推荐)
使用 @DependsOn 确保 BIRedisUtil 在其他 Bean 之后初始化:
@Component
@DependsOn("biRedisTemplate") // 依赖 biRedisTemplate 先初始化
public class BIRedisUtil { ... }
- 使用 @PostConstruct 初始化静态变量
private static RedisTemplate redisTemplate;
@Autowired
@Qualifier("biRedisTemplate")
private RedisTemplate _redisTemplate; // 实例变量接收注入@PostConstruct
public void init() {BIRedisUtil.redisTemplate = this._redisTemplate; // 转赋给静态变量
}
四、总结
核心原因是 Spring 不直接支持静态变量注入,导致依赖注入的时序不可控。服务器环境因复杂的初始化流程和高并发场景,更容易暴露此问题。改用实例变量是根本解决方案。