美文网首页开放平台
Redis分布式锁防止超卖

Redis分布式锁防止超卖

作者: 狗没拿伞 | 来源:发表于2018-07-04 20:37 被阅读100次

 SETNX ,命令解释http://redis.cn/commands/setnx.html,GETSET,命令解释http://redis.cn/commands/getset.html

public class RedisLock {

    @Autowired
    private StringRedisTemplate redisTemplate;

    /**
    *@Description
    *@Date 14:10 2018/6/17
    *@Param value:当前时间+超时时间
    *@return
    **/
    public  boolean lock(String key,String value){
        if (redisTemplate.opsForValue().setIfAbsent(key,value)){
            return true;
        }

//下面代码解决了 死锁问题 和 保证只有一个线程拿锁。
/**1.死锁 。当相关代码之前上锁后,如果执行代码的过程中抛错了,
那么就会出现没有解锁的问题,后续线程无法获得锁,也就出现了死锁的问题。
而以下代码后有返回true的可能,也就解决了死锁问题。
*/
/**2.保证只有一个线程拿锁。比如两个线程同时进入下面代码,并且其value都为B,currentValue = A。
一条代码只可能一个线程执行。
当第一个线程执行时,oldvalue = A,如果符合if条件成功拿锁。
当第二个执行时oldValue就已经是B了,所以保证了只有一个线程拿锁。
*/

        String currentValue = redisTemplate.opsForValue().get(key);
        //如果锁过期
        if (!StringUtils.isEmpty(currentValue) && Long.parseLong(currentValue) < System.currentTimeMillis()){
            //获取上一个锁的时间
            String oldValue = redisTemplate.opsForValue().getAndSet(key,value);
            if (!StringUtils.isEmpty(oldValue) && oldValue.equals(currentValue)){
                return true;
            }
        }

        return false;
    }

    /**
    *@Author banrityan
    *@Description 解锁
    *@Date 14:25 2018/6/17
    *@Param
    *@return
    **/
    public void unlock(String key,String value){
        try{
            String currentValue = redisTemplate.opsForValue().get(key);
            if (!StringUtils.isEmpty(currentValue) && currentValue.equals(value)){
                redisTemplate.opsForValue().getOperations().delete(key);
            }
        }catch (Exception e){
            log.error("【redis分布式锁】解锁异常,{}",e);
        }
    }
}

 解释在代码注释中。
 然后值需要在要执行的代码前加锁

        //加锁
        long time = System.currentTimeMillis() + TIMEOUT;
        if (!redisLock.lock(productId,String.valueOf(time))){
            throw new SellException(101,"哎哟喂,人太多,请稍后再试~~");
        }

 然后执行完后记得解锁

 //解锁
        redisLock.unlock(productId,String.valueOf(time));

内容如上。\( ̄︶ ̄)/

参考资料:无所不知的度娘+各位大佬的博客

相关文章

  • 81 zookeeper 分布式锁

    1,分布式锁实现本质? 2,分布式有哪些应用场景? 2,用户下单库存超卖问题,防止超卖Redis解决。 3,锁的特...

  • php+redis消息队列实现抢购功能

    实现功能: 基于redis队列,防止高并发的超卖 基于mysql的事务加排它锁,防止高并发的超卖基于redis队列...

  • Redis分布式锁防止超卖

     SETNX ,命令解释http://redis.cn/commands/setnx.html,GETSET,命令...

  • Redis分布式锁的实战应用

    以商品超卖为例讲解Redis分布式锁 主要讲解Redis实现分布式锁的两种实现方式:Jedis实现、Redisso...

  • go-zero基础组件-分布式锁RedisLock

    为什么需要分布式锁 用户下单 锁住uid,防止重复下单。 库存扣减 锁住库存,防止超卖。 余额扣减 锁住账户,防止...

  • 用 Go + Redis 实现分布式锁

    为什么需要分布式锁 用户下单 锁住 uid,防止重复下单。 库存扣减 锁住库存,防止超卖。 余额扣减 锁住账户,防...

  • redisson中的看门狗机制总结

    1:普通的Redis分布式锁的缺陷 我们在网上看到的redis分布式锁的工具方法,大都满足互斥、防止死锁的特性,有...

  • redis 面试题

    技巧:1、redis + lua 解决高并发场景下的写操作2、redis 分布式锁,防止并发写3、redis 队列...

  • 大佬浅谈分布式锁

    redis 实现 redis 分布锁一、redis 实现分布式锁(可重入锁)redission 实现分布式锁1、对...

  • redis笔记

    redis使用项目经验 验证码 热点数据 使用场景 事务执行流程 事务注意事项 watch 分布式锁(解决超卖问题...

网友评论

    本文标题:Redis分布式锁防止超卖

    本文链接:https://www.haomeiwen.com/subject/gfdyuftx.html