1、使用Redis作为分布式锁的原子性问题
原方案:
① SETNX $LOCK_BUSI_KEY $REQ_ID
② EXPIRE $LOCK_BUSI_KEY $LOCK_TIME
问题:
使用SETNX,如果锁不存在,则SET成功,返回1;否则,返回0。
为了保证锁在异常退出时,仍能超时释放,使用了EXPIRE;但是由于①和②为非原子操作,导致EXPIRE未能设置成功,造成死锁。
修复方案:
Redis从2.6.12版本开始,在SET命令中增加了EX选项支持设置过期时间,NX参数设置KEY不存在时才写入值
SET $LOCK_BUSI_KEY $REQ_ID EX $LOCK_TIME NX
2、Redis的hset导致hgetAll数据不完整的问题
问题:
将config写入DB,业务获取所有的config时加载到Redis缓存,放入set中。即service.getAll执行Redis.hgetAll加载数据,若未命中,则从DB加载到Redis中。
另外,在业务更新config时,允许单条更新,DB更新成功后,单独hset刷新到Redis缓存。或者service允许get单条数据,未命中缓存时,从DB加载单条hset到Redis。
问题症状:
会导致hgetAll返回的数据与DB数据不一致,可能比DB的条数少。
修复方案:
不允许单独执行hset局部刷新缓存的方法。