redis.clients.jedis.exceptions.JedisDataException: ERR hash value is not an integer

java 专栏收录该内容
48 篇文章 0 订阅

业务场景

https://blog.csdn.net/qq_31071543/article/details/84752778

推测抛出异常的原因
第一次访问接口时通过void setEntryInHash(String key,String field,String vlaue)方法在redis中存放用户访问当前接口的次数为1,此时不会抛出异常,通过redis命令查看存放的数据为
field为: “��” value为:"��。第二次访问接口时通过void incrementValue(String key,String field,int increment)方法改变用户访问接口的次数,此时就会抛出异常。由此可以看出抛出异常的原因可能是因为第一次存放的数据已经发生乱码,第二次通过传入一个int去改变内存中用户访问的接口次数就会发生乱码。

分析抛出异常的原因
在项目中采用 Spring Data Redis操作数据库缓存,通过RedisTemplate模版来操作redis数据库。RedisTemplatebean配置为:

@Bean
	RedisTemplate<String, Object> template(RedisConnectionFactory connectionFactory) {
		RedisTemplate<String, Object> template = new RedisTemplate<String, Object>();
		template.setConnectionFactory(connectionFactory);
		// 定义value的序列化方式
		Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(Object.class);
		ObjectMapper om = new ObjectMapper();
		om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
		om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
		jackson2JsonRedisSerializer.setObjectMapper(om);
		template.setValueSerializer(jackson2JsonRedisSerializer);
		template.setKeySerializer(new StringRedisSerializer(Charset.forName("UTF-8")));
		template.afterPropertiesSet();
		return template;
	}

发现没有配置hashkey,hashValue的序列化方式,因此当我们采用RedisTemplate操作hash数据结构,就会采用默认的序列化方式JdkSerializationRedisSerializer(这种序列化方式主要用来操作对象,操作字符串就会发生乱码的情况)。

解决方案
方案一:
在使用注解@Bean返回RedisTemplate的时候,同时配置hashKey与hashValue的序列化方式。

@Bean
	RedisTemplate<String, Object> template(RedisConnectionFactory connectionFactory) {
		RedisTemplate<String, Object> template = new RedisTemplate<String, Object>();
		template.setConnectionFactory(connectionFactory);
		// 定义value的序列化方式
		Jackson2JsonRedisSerializer<Object> jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer<Object>(Object.class);
		ObjectMapper om = new ObjectMapper();
		om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
		om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL);
		jackson2JsonRedisSerializer.setObjectMapper(om);
		template.setValueSerializer(jackson2JsonRedisSerializer);
		template.setKeySerializer(new StringRedisSerializer(Charset.forName("UTF-8")));
		//save hash use StringRedisSerializer as serial method
		template.setHashKeySerializer(new StringRedisSerializer(Charset.forName("UTF-8")));
		template.setHashValueSerializer(new StringRedisSerializer(Charset.forName("UTF-8")));
		template.afterPropertiesSet();
		return template;
	}

方案二:
采用Spring Data Redis的StringRedisTemplate这个模版来操作数据。

@Slf4j
@Service("redisHashService")
public class RedisHashServiceImpl implements RedisHashService {
	
	
	@Autowired
	StringRedisTemplate redisTemplate;
	
	@Resource(name="redisTemplate")
	HashOperations<String, Object, Object>  opsForHash;

	@Override
	public Object getValueByKeyAndField(String key, String field) {
		return opsForHash.get(key, field);
	}

	@Override
	public boolean hasKey(String key) {
		return redisTemplate.hasKey(key);
	}

	@Override
	public void setExpire(String key, long during, TimeUnit timeUnit) {
		redisTemplate.expire(key, during, timeUnit);
	}

	@Override
	public boolean hasKeyAndField(String key, String field) {
		return opsForHash.hasKey(key, field);
	}

	@Override
	public void setEntryInHash(String key, String field, String value) {
		 opsForHash.put(key, field, value);
	}

	@Override
	public void incrementValue(String key, String field,long incredment) {
		opsForHash.increment(key, field, incredment);
	}

}
  • 1
    点赞
  • 0
    评论
  • 0
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

©️2021 CSDN 皮肤主题: 编程工作室 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值