使用Redisd的RedisTemplate类中调用自增方法increment出现的报错问题

报错代码片段
    @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中有多种序列化格式,需要根据自己需求选择对应的序列化格式

  1. GenericJackson2JsonRedisSerializer、Jackson2JsonRedisSerializer是先将对象转为json,然后再保存到redis,所以,1在redis中是字符串1,所以无法进行加1。
  2. JdkSerializationRedisSerializer使用的jdk对象序列化,序列化后的值有类信息、版本号等,所以是一个包含很多字母的字符串,所以根本无法加1。
  3. GenericToStringSerializer、StringRedisSerializer将字符串的值直接转为字节数组,所以保存到redis中是数字,所以可以进行加1

ToStringSerializer、StringRedisSerializer将字符串的值直接转为字节数组,所以保存到redis中是数字,所以可以进行加1

参考文章内容:https://juejin.cn/post/6844903681926791182

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
如果在使用 Spring Data Redis 的 `increment` 方法进行自增操作时,出现了 `ERR value is not an integer or out of range` 的错误,可能是因为 Redis 中对应的键存储的值不是整数类型,或者存储的整数值超出了 Redis 数据类型的范围。 为了解决这个问题,你可以在调用 `increment` 方法之前,先使用 RedisTemplate 的 `type` 方法获取 Redis 中对应键的数据类型,如果是字符串类型,就尝试将其转换为整数类型。如果转换失败,就抛出异常或者返回错误信息,避免出现 `ERR value is not an integer or out of range` 的错误。 示例代码如下: ```java @Autowired private RedisTemplate<String, Object> redisTemplate; public Long incrementCounter(String key, long delta) { DataType dataType = redisTemplate.type(key); if (dataType == DataType.STRING) { Object value = redisTemplate.opsForValue().get(key); if (value == null) { redisTemplate.opsForValue().set(key, delta); return delta; } else { try { Long longValue = Long.parseLong(value.toString()); return redisTemplate.opsForValue().increment(key, delta); } catch (NumberFormatException e) { throw new RuntimeException("Value is not an integer: " + value); } } } else { throw new RuntimeException("Key is not a string: " + key); } } ``` 上述代码中,`redisTemplate` 是一个已经通过 Spring 配置注入的 RedisTemplate 对象,泛型参数 `<String, Object>` 表示 Redis 中键名和键值的数据类型。`incrementCounter` 方法接受一个键名和自增步长,首先使用 `type` 方法获取 Redis 中对应键的数据类型,如果是字符串类型,就尝试将其转换为整数类型。如果转换失败,就抛出异常或者返回错误信息;如果转换成功,就调用 `increment` 方法实现自增操作,并返回自增后的结果。如果键名对应的数据类型不是字符串类型,也会抛出异常或者返回错误信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值