在springboot整合Redis的时候 ,项目无法启动,报:
Field redisTemplate in com.xzh.AccessLimitController required a bean of type 'org.springframework.data.redis.core.RedisTemplate' that could not be found.
我的RedisTemplate注入如下:
@Autowired
private RedisTemplate<String,Integer> redisTemplate;
RedisConfig如下:
@Configuration
@EnableCaching
public class RedisConfig {
@Bean
public CacheManager cacheManager(RedisConnectionFactory redisConnectionFactory) {
// 默认Redis缓存过期时间为1小时50分钟
RedisCacheConfiguration redisCacheConfiguration =
RedisCacheConfiguration.defaultCacheConfig()
.entryTtl(Duration.ofMinutes(110));
return RedisCacheManager
.builder(RedisCacheWriter.nonLockingRedisCacheWriter(redisConnectionFactory))
.cacheDefaults(redisCacheConfiguration).build();
}
@Bean
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory) {
RedisTemplate<Object, Object> template = new RedisTemplate<>();
template.setConnectionFactory(connectionFactory);
//使用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值(默认使用JDK的序列化方式)
Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper mapper = new ObjectMapper();
mapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
mapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
serializer.setObjectMapper(mapper);
template.setValueSerializer(serializer);
//使用StringRedisSerializer来序列化和反序列化redis的key值
template.setKeySerializer(new StringRedisSerializer());
template.afterPropertiesSet();
return template;
}
@Bean
public StringRedisTemplate stringRedisTemplate(RedisConnectionFactory factory) {
StringRedisTemplate stringRedisTemplate = new StringRedisTemplate();
stringRedisTemplate.setConnectionFactory(factory);
return stringRedisTemplate;
}
奇怪的是,不带泛型的RedisTemplate是可以正常注入的。在网上查找解决方法,发现,将 @Autowired 换成 @Resource 就可以了
@Resource
private RedisTemplate<String, Integer> redisTemplate;
试了一下,果然成功啦。那么问题来了, @Autowired 和 @Resource 到底有啥区别呢?
请看:@Autowired 和 @Resource 的区别?
因为@Autowired是根据类型来注入的,泛型也被考虑进去了,显然,我们的RedisTemplate里并没有我想要注入的这一类型,所以报错了;@Resource默认根据名称注入,redisTemplate 这个bean在我们的RedisConfig里被加到了spring中,所以是可以被注入的。
既然知道了rootcause,那么解决方法就很多了,一定要使用@Autowired也是可以的,只要spring容器中有我们需要的类型的RedisTemple就可以啦:
@Bean
public RedisTemplate<String, Integer> redisTemplate(RedisConnectionFactory connectionFactory) {
RedisTemplate<String, Integer> template = new RedisTemplate<>();
template.setConnectionFactory(connectionFactory);
//使用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值(默认使用JDK的序列化方式)
Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(String.class);
ObjectMapper mapper = new ObjectMapper();
mapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
mapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
serializer.setObjectMapper(mapper);
template.setValueSerializer(serializer);
//使用StringRedisSerializer来序列化和反序列化redis的key值
template.setKeySerializer(new StringRedisSerializer());
template.afterPropertiesSet();
return template;
}
有的同学可能要说啦,用@Autowired 配合@Qualifier 使用,不也是按名称注入的吗,但是忽略了一点是,虽然这样使用它会按名称找bean,但是它还是有bean 类型的限制的,看下面的提示就知道啦:
事实证明,这样是OK的,所以解决问题最重要的一个环节就是找到造成问题的原因,一旦知道了,解决起来就很简单了。