场景
今天写业务代码,先从redis中读取值,不存在的话加redis锁,锁定之后再次尝试读取一次redis,没有值的话去db中读取并写redis缓存。
非常简单的代码,确报错WRONGTYPE Operation against a key holding the wrong kind of value
分析
错误发生在锁定之后再次读取redis的时候,报错的字面意思很简单,key值类型不对,但是理论上这个时候不是还没有这个缓存key吗,就算有,也是其他线程或者用户实例给设置的缓存,怎么可能报错key类型不对。照着源码排查了我十几分钟无果,直到我看了下我的业务代码:
@Override
public List<ImagePoolDto> listImagePoolAndImage(VrImagePoolListQueryVo vo) throws Exception {
// 从缓存中获取
String cacheKey = VrImagePool.cacheKey(vo.getParentId(), vo.getTitle());
List<ImagePoolDto> cache = this.redisService.getListFromRedis(cacheKey, ImagePoolDto.class);
if (null != cache) {
return cache;
} else {
// 缓存中没有,从db中获取并写缓存
AtomicBoolean cacheFlag = new AtomicBoolean(false);
AtomicReference<List<ImagePoolDto>> cacheRef = new AtomicReference<>

在执行业务代码时,尝试从Redis中读取值,若不存在则加锁并在再次尝试读取前从DB中获取并写入缓存。然而在加锁后读取Redis时遇到WRONGTYPE错误,表示键的类型不正确。通过查看源码并未发现问题,最后发现是由于在获取锁时使用了缓存键,导致锁创建的同时在Redis中创建了键,但其类型不匹配。修复代码后问题解决。
最低0.47元/天 解锁文章
954





