可重入锁实例(ReentrantLock)-商品订购

@RequestMapping("/test")
@RestController
public class TestController {

    ConcurrentHashMap<String, Goods> goodsMap = new ConcurrentHashMap<>();

    /**
     * 下单
     *
     * @return
     */
    @RequestMapping("/placeOrder")
    public ResponseEntity test(String id) {
        System.out.println("正在下单:商品主键为:" + id);
        //初始化商品
        if (goodsMap.get(id) == null) {
            return ResponseEntity.status(500).body("下单商品不存在!");
        } else {
            return goodsMap.get(id).placeOrder();

        }
    }

    @RequestMapping("init")
    @PostConstruct
    public ResponseEntity init() {
        try {
            //随机创建商品
            Goods goods = new Goods();
            goods.setCount(ThreadLocalRandom.current().nextInt(100));
            goodsMap.put(Thread.currentThread().getId() + "", goods);
            System.out.println("商品主键:" + Thread.currentThread().getId() + "商品初始化数量为:" + goods.getCount());
        } catch (Exception e) {
            e.printStackTrace();
            return ResponseEntity.status(500).body("初始化库存失败");
        }
        return ResponseEntity.ok("初始化成功" + JSON.toJSONString(goodsMap));
    }

    class Goods {
        /**
         * 令牌 -- 可重入锁
         */
        ReentrantLock lock = new ReentrantLock();
        private Integer count = 0;//共享变量 库存

        /**
         * 用于模拟数据库查询延迟
         *
         * @return
         * @throws InterruptedException
         */
        public Integer getCount() throws InterruptedException {
            Thread.sleep(ThreadLocalRandom.current().nextInt(200)); //模拟数据库延迟
            return count;
        }

        /**
         * 用于模拟数据库写入延迟
         *
         * @return
         * @throws InterruptedException
         */
        public void setCount(Integer count) throws InterruptedException {
            Thread.sleep(ThreadLocalRandom.current().nextInt(200)); //模拟数据库延迟
            this.count = count;
        }

        /**
         * 出库
         *
         * @throws InterruptedException
         */
        public void stockRemoval() throws InterruptedException {
            this.setCount(this.getCount() - 1);
            System.out.println(Thread.currentThread().getName() + "下单成功!剩余数量" + getCount());
        }

        /**
         * 商品下单
         *
         * @return
         */
        ResponseEntity placeOrder() {
            //判断当前线程是否是持有令牌者
            //进来先锁住线程 避免资源共享
            //尝试获取锁 如果为TRUE 表示锁未被占用 如果为False 则锁已经被其他线程占用
            if (lock.tryLock()) {
                try {
                    //判断当前库存是否大于0
                    Integer count = getCount();
                    System.out.println("正在下单 当前库存为" + count);
                    if (count > 0) {
                        //商品出库
                        stockRemoval();
                        // Thread.sleep(3000);  //模仿处理时间 锁死3秒 把其他的线程排除在外
                        return ResponseEntity.ok("下单成功!");
                    } else {
                        return ResponseEntity.status(500).body("库存不足!");
                    }
                } catch (Exception exception) {
                    exception.printStackTrace();
                    return ResponseEntity.status(500).body("系统异常");
                } finally {
                    //操作到最后都要释放锁 避免死锁
                    lock.unlock();
                }
            } else {
                //未持有线程令牌者 直接响应 避免排队等待
                return ResponseEntity.status(500).body("下单人数过多,可以等待一段时间再下单");
            }
        }
    }
}
拒绝线程等待,增加用户体验
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值