redis分布式锁

 1  缓存

当执行增删改操纵时必须保证缓存和数据库数据一致性。---删除缓存


    @Override
    public User insert(User user) {
        userMapper.insert(user);
        return user;
    }

    @Override
    public Integer delete(Integer id) {
        redisTemplate.delete("user::" + id);
        int i = userMapper.deleteById(id);
        return i;
    }

    @Override
    public User update(User user) {
        redisTemplate.delete("user::" + user.getId());
        userMapper.updateById(user);
        return user;
    }

2   redis使用分布式锁

(1)通过使用jmeter压测工具测试

同一个库存数被多个线程卖,线程安全问题。---思考:之间出现线程安全问题时如何解决。

可以使用锁解决:----synchronized和Lock锁

    @Autowired
    private StockDao stockDao;
   
    Lock lock = new ReentrantLock();

    public String Test(Integer id){
        try {
            lock.lock(); //  加锁
            Stock stock = stockDao.selectById(id);  //  根据id查询
            if (stock.getNum() > 0){     //  如果对象的num值大于0    代表还有
                stock.setNum(stock.getNum() - 1);      //  修改num值(括号里是把num值减一)
                stockDao.updateById(stock);          //   然后修改数据库
                System.out.println("库存剩余数量:" + stock.getNum());    //  控制台输出num的数量
                return "减库存成功";           //  给页面发送---->减库存成功
            }else {          //  代表对象的num值不大于0(没有了)
                System.out.println("库存不足");    //   控制台输出库存不足
                return "库存减失败";      //   给页面发送---->库存减失败
            }
        }finally {   //  try   finally必进
            lock.unlock();  //  释放锁
        }
    }

上面的synchronized或Lock锁是否适合集群模式|分布式系统。不适合、因为synchronized都是基于JVM的本地锁。

 需要在idea中跑项目的集群

 配置nginx 

 启动nginx

 jmeter压测

 两台集群出现重卖问题。

2.2   使用redis来解决分布式锁

 

    @Autowired
    private StockDao stockDao;

    @Autowired
    private StringRedisTemplate redisTemplate;

    public String test2(Integer id){
        ValueOperations<String, String> forValue = redisTemplate.opsForValue();  // 获取String类型的
        //  线程占锁  占到的返回true并保存到缓存中30秒   没占到返回false
        Boolean aBoolean = forValue.setIfAbsent("product::" + id, "", 30, TimeUnit.SECONDS);
        while (!aBoolean){     // 判断如果为false
            try {
                Thread.sleep(20);    // 线程睡眠20ms  醒来继续抢锁
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        try {
            Stock stock = stockDao.selectById(id);   //  根据id查询
            if (stock.getNum() > 0){      //  如果对象的num值大于0    代表还有
                stock.setNum(stock.getNum() - 1);     //  修改num值(括号里是把num值减一)
                stockDao.updateById(stock);          //   然后修改数据库
                System.out.println("库存剩余数量:" + stock.getNum());    //  控制台输出num的数量
                return "减库存成功";          //  给页面发送---->减库存成功
            }else {               //  代表对象的num值不大于0(没有了)
                System.out.println("库存不足");      //   控制台输出库存不足
                return "库存减失败";      //   给页面发送---->库存减失败
            }
        }finally {
            //释放锁资源
            redisTemplate.delete("product::"+ id);
        }
    }

如果你的业务代码的执行时间超过30s,当前线程删除的是其他线程的锁资源。 --watchDog机制

每个10s检测当前线程是否还持有所资源,如果持有则为当前线程延迟。---可以自己设置watchDog机制,---第三方Redission完美的解决分布式锁

2.3  redisson完美解决redis超时问题

<dependency>
     <groupId>org.redisson</groupId>
     <artifactId>redisson</artifactId>
     <version>3.13.4</version>
</dependency>

 (2)main函数

@Bean //创建redisson交于spring容器来管理
    public RedissonClient redisson() {
        Config config = new Config();
        config.useSingleServer().setAddress("redis://192.168.40.166:6379");
        RedissonClient redisson = Redisson.create(config);
        return redisson;
    }

(3)使用

    @Autowired
    private StockDao stockDao;

    @Autowired
    private RedissonClient redisson;

    public  String jianStock(Integer pid){
        RLock lock = redisson.getLock("product::" + pid);
        try {
            lock.lock(30,TimeUnit.SECONDS);//加锁: 如果程序执行是出现一次
            //1. 查询指定的商品库存
            Stock stock = stockDao.selectById(pid);
            if (stock.getNum() > 0) {
                //2.库存减1
                stock.setNum(stock.getNum() - 1);
                stockDao.updateById(stock);
                System.out.println("库存剩余数量:" + stock.getNum());
                return "减库存成功";
            } else {
                System.out.println("库存不足");
                return "库存减失败";
            }
        }
        finally {
            lock.unlock();
        }
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值