IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> 【Java】高并发下如何防止库存超发 -> 正文阅读

[大数据]【Java】高并发下如何防止库存超发

知识补充

乐观锁: 读的时候不加锁,写的时候认为别的线程是不会修改数据的。如果别的线程修改了数据,放弃本线程的修改,重新进行尝试数据修改(CAS)。

悲观锁: 修改的数据的时候,认为别的线程一定会修改数据,直接将数据锁死,直到修改完数据。

一、什么时候扣库存

比较合理的方式是,用户提交订单后扣减库存。当用户超时没有进行支付的时候,系统将提交的订单取消,并进行订单的回退。

二、怎么防止用户重复点击

1、前端处理:用户点击后,将按钮置灰。
2、后端处理:用redis统计用户调接口的顺序,调用次数过多,直接返回失败。如果调用次数很不合理,直接拉黑。

三、怎么安全的扣减库存

方案一、数据库操作商品库存采用乐观锁防止超卖

  • 实现
update sku_stock set stock = stock - num where sku_code = '' and stock - num > 0;
  • 分析

并发场景下,两个请求同时过来,数据会保证只有一个用户扣减库存成功。在并发量不是很大的情况下可以这么处理,当并发量特别大的时候,很有可能会把数据库打垮。

方案二、使用Redis,强制把多线程转成单线程处理

  • 实现
/**
 * 缺点并发不高,同时只能一个用户抢占操作,用户体验不好!
 *
 * @param orderSkuAo
 */
public boolean subtractStock(OrderSkuAo orderSkuAo) {
    String lockKey = "shop-product-stock-subtract" + orderSkuAo.getOrderCode();
    if (redis.get(lockKey)) {
        return false;
    }
    try {
    	//加锁
        lock.lock(lockKey, 1L, 10L);
        //处理逻辑
    } catch (Exception e) {
        LogUtil.error("e=", e);
    } finally {
    	//解锁
        lock.unLock(lockKey);
    }
    return true;
}
  • 分析

利用redis的分布式锁,可以将抢购的场景强制转换成单线程操作。缺点是并发也不高,处理相对比较慢,不太适合高并发,抢购的场景。但是与方案一相比,减轻了数据库的压力。

方案三、redis + mq + mysql 保证库存安全,满足高并发处理

  • 实现
/**
 * 扣库存操作,秒杀的处理方案
 *
 * @param orderCode
 * @param skuCode
 * @param num
 * @return
 */
public boolean subtractStock(String orderCode, String skuCode, Integer num) {
    String key = "shop-product-stock" + skuCode;
    Object value = redis.get(key);
    if (value == null) {
        //前提 提前将商品库存放入缓存 ,如果缓存不存在,视为没有该商品
        return false;
    }
    
    //先检查 库存是否充足
    Integer stock = (Integer) value;
    if (stock < num) {
        LogUtil.info("库存不足");
        return false;
    }
    
    //不可在这里直接操作数据库减库存,否则导致数据不安全
    //因为此时可能有其他线程已经将redis的key修改了
    //redis 减少库存,然后才能操作数据库
    Long newStock = redis.increment(key, -num.longValue());
    //库存充足
    if (newStock >= 0) {
        LogUtil.info("成功抢购");
        //TODO 真正扣库存操作 可用MQ 进行 redis 和 mysql 的数据同步,减少响应时间
    } else {
        //库存不足,需要增加刚刚减去的库存
        redis.increment(key, num.longValue());
        LogUtil.info("库存不足,并发");
        return false;
    }
    
    return true;
}
  • 分析

利用 redis.increment 的原子操作,可以保证库存安全,利用MQ可以保证高并发的响应时间。但是需要保证商品的库存信息在redis里面,并且保证 Redis 和 MySQL的数据同步。

参考链接: 大佬的博客

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2022-06-23 00:55:58  更:2022-06-23 00:56:40 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/20 0:32:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码