报错代码片段
@Resource
private RedisTemplate<String,String > redisTemplate;
//键值对中的value以String数据类型存储
redisTemplate.opsForValue().set(uploadedNoKey, "1");
//根据key实现value的自增
redisTemplate.opsForValue().increment(uploadedNoKey);
控制台抛出的报错信息:
nested exception is redis.clients.jedis.exceptions.JedisDataException: ERR value is not an integer or out of range
查找原因
Redis在未序列化存入内存中是默认的序列化格式EMBEDDED (内置的序列化格式),用于将对象序列化为字节数组。Redis根据存储的对象类型选择不同的序列化策略。例如,字符串对象使用简单的字符串编码,列表对象使用快速列表编码,哈希对象使用压缩列表或哈希表编码等,查看到的value值不是存入的字符串 “1”,而是进行一列处理后的编码。
为此在调用increment实现自增中,key对应的值不为数字,不能进行自增操作。虽然以字符串 “1” 的形式存入,但是在指定格式的序列化之后存到redis中是数字,所以可以进行加1。
序列化前存入内存的 key-value
解决方法
对Redis存入内存的键值对 key - value 进行序列化
@Bean
public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory connectionFactory) {
RedisTemplate<String, Object> template = new RedisTemplate<>();
GenericToStringSerializer genericToStringSerializer = new GenericToStringSerializer<>(Object.class);
template.setConnectionFactory(connectionFactory);
// key采用String的序列化方式
template.setKeySerializer(new StringRedisSerializer());
// value序列化方式采用jackson
template.setValueSerializer(genericToStringSerializer);
// hash的key也采用String的序列化方式
template.setHashKeySerializer(new StringRedisSerializer());
// hash的value序列化方式采用jackson
template.setHashValueSerializer(genericToStringSerializer);
template.afterPropertiesSet();
return template;
}
}
序列化后存入内存的 key - value
备注
Redis中有多种序列化格式,需要根据自己需求选择对应的序列化格式
- GenericJackson2JsonRedisSerializer、Jackson2JsonRedisSerializer是先将对象转为json,然后再保存到redis,所以,1在redis中是字符串1,所以无法进行加1。
- JdkSerializationRedisSerializer使用的jdk对象序列化,序列化后的值有类信息、版本号等,所以是一个包含很多字母的字符串,所以根本无法加1。
- GenericToStringSerializer、StringRedisSerializer将字符串的值直接转为字节数组,所以保存到redis中是数字,所以可以进行加1
ToStringSerializer、StringRedisSerializer将字符串的值直接转为字节数组,所以保存到redis中是数字,所以可以进行加1
参考文章内容:https://juejin.cn/post/6844903681926791182