redis锁处理并发问题

转载:https://blog.csdn.net/u010963948/article/details/79381191

redis锁处理并发问题

redis锁处理高并发问题十分常见,使用的时候常见有几种错误,和对应的解决办法,在此进行自己的总结和整理。

  • set方式
  • setnx方式
  • setnx+getset方式

set方式 
作为redis小白,一开始能想到的使用redis加锁的方式就是set。 
加锁:redis中set一个值,set(lock,1); 
并发处理:其他线程必须拿到这个值,才可以往下进行,否则等待。

while(jedis.exists(lock)){
        Thread.sleep(500);
    }
    set(lock1);
    执行业务代码;
    jedis.del(lock);

释放锁:执行完业务代码之后,释放redis锁,jedis.del(lock) 
防止死锁:set(lock,1) —>3秒后未释放,则自动释放setex(lock, 3, 1) 
问题:高并发情况下,进程同时获取锁状态为null,同时设置,锁之间相互覆盖,但是俩进程仍在并发执行业务代码。 
setnx方式 
后来发现有setnx的原子操作命令,锁存在不能设置值,返回0;锁不存在,则设置锁,返回1; 
加锁:jedis.setnx(lock, 1) 
并发处理:

while(jedis.setnx(lock,1)==0){
    Thread.sleep(300);
}
执行业务代码;
jedis.del(lock);

释放锁:执行完业务代码之后,释放redis锁,jedis.del(lock) 
问题:当进程执行出现问题,锁未释放,则其他进程永远处于阻塞状态,出现死锁。 
防止死锁:加锁时带上时间戳,setnx(lock, 时间戳+超时时间)

while(jedis.setnx(lock,now+超时时间)==0){
        if(jedis.get(lock)<now){
            jedis.del(lock);
            jedis.setnx(lock,now+超时时间);
            break;
        }else{
            Thread.sleep(300);
        }
    }
    执行业务代码;
    jedis.del(lock);

问题:当俩进程同时读到发现锁超时,都去释放锁,相互覆盖,则俩进程同时获得锁,仍并发执行业务代码。 
setnx+getset方式 
为解决上面的问题,可以使用getset命令,getset设置键值,并返回原来的键值。 
加锁:setnx(lock, 时间戳+超时时间) 
解决并发:

while(jedis.setnx(lock, now+超时时间)==0){
    if(now>jedis.get(lock) && now>jedis.getset(lock, now+超时时间)){
        break;
    }else{
        Thread.sleep(300);
    }
}
执行业务代码;
jedis.del(lock);

释放锁:jedis.del(lock);


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值