秒殺系統如何保證資料庫不崩潰以及防止商品超賣

来源:https://www.cnblogs.com/jae-tech/archive/2022/07/18/16487925.html
-Advertisement-
Play Games

1、應用場景 電商商城,商家上架了一個秒殺活動,早上10點開始,商品A參與秒殺,一共有20個庫存,預計10W的人去搶。 2、面臨問題 高併發、庫存不可超賣 3、問題解決 1)高併發,我們不能把所有的請求都去資料庫查商品詳情,查商品庫存,這樣資料庫會頂不住,很容易的我們就想到了用Redis解決; 2) ...


1、應用場景

電商商城,商家上架了一個秒殺活動,早上10點開始,商品A參與秒殺,一共有20個庫存,預計10W的人去搶。

 

2、面臨問題

高併發、庫存不可超賣

 

3、問題解決

1)高併發,我們不能把所有的請求都去資料庫查商品詳情,查商品庫存,這樣資料庫會頂不住,很容易的我們就想到了用Redis解決;

2)庫存超賣問題,這個問題主要是由於用戶在同時讀取到的庫存均為大於0,從而認為我們該商品還沒被秒完,繼續創建了訂單,導致了商品超賣了。 

 

4、編碼實現  

1、資料庫新建兩張表

秒殺訂單

CREATE TABLE `ms_order` (
  `ms_order_id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '訂單ID',
  `created_time` datetime DEFAULT NULL COMMENT '創建時間',
  `order_price` decimal(12,2) DEFAULT NULL COMMENT '訂單總價',
  `state` tinyint(1) DEFAULT '1' COMMENT '訂單狀態 1未支付 2已支付 3已發貨 4已收貨 -1已取消',
  `pay_time` datetime DEFAULT NULL COMMENT '支付時間',
  `fh_time` datetime DEFAULT NULL COMMENT '發貨時間',
  PRIMARY KEY (`ms_order_id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COMMENT='秒殺訂單';

 

秒殺商品

CREATE TABLE `ms_product` (
  `ms_product_id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '秒殺商品ID',
  `product_name` varchar(100) DEFAULT NULL COMMENT '商品名稱',
  `origin_price` decimal(12,2) DEFAULT NULL COMMENT '商品原價',
  `ms_price` decimal(12,2) DEFAULT NULL COMMENT '秒殺價',
  `product_img` varchar(255) DEFAULT NULL COMMENT '商品圖片',
  `state` tinyint(1) DEFAULT NULL COMMENT '商品狀態 1已上架 -1已下架',
  `product_summary` varchar(255) DEFAULT NULL COMMENT '商品描述',
  `product_details` text COMMENT '商品詳情',
 PRIMARY KEY (`ms_product_id`) 
) ENGINE
=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COMMENT='秒殺商品';

 

2、設置商品庫存,正式的流程肯定是由後臺添加商品時初始化,這邊為了方便,直接用Redis可視化工具插入了商品,秒殺商品ID為1的設置20個庫存,同時資料庫也要設置20個庫存,利於我們分析扣減庫存是否一致

 

 

3、敲代碼

1)寫一個下單介面

@PostMapping(value = "/add")
    public ResultMsg add(HttpServletRequest request, MsOrder msOrder,Long ms_product_id) {
        String interfaceName = "下單測試";
        try {
            User user = getUser();
            return new ResultMsg(true, msOrderService.insert(msOrder, user,ms_product_id));
        } catch (ServiceRuntimeException e) {
            return fail(e);
        } catch (Exception e) {
            return error(interfaceName, e, request);
        }
    }

 

2)邏輯處理

利用lua腳本減庫存,lua腳本如下

local isExist = redis.call('exists', KEYS[1]);
if (tonumber(isExist) > 0) then
    local goodsNumber = redis.call('get', KEYS[1]);
    if (tonumber(goodsNumber) > 0) then
        redis.call('decr',KEYS[1]);
        return 1;
    else
        redis.call('del', KEYS[1]);
        return 0;
        end;
else
return -1;
end;

 

lua配置類

@Configuration
public class LuaConfiguration {
    @Bean
    public DefaultRedisScript<Long> redisScript() {
        DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();
        redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource("script/Stock.lua")));
        redisScript.setResultType(Long.class);
        return redisScript;
    }

}

 

扣減Redis中對應的商品庫存

@Component
public class LuaReduceStock {

    @Resource
    private DefaultRedisScript<Long> redisScript;
    @Resource
    private StringRedisTemplate stringRedisTemplate;

    /**
     * 減庫存
     * @param key
     * @return
     */
    public boolean reduceStock(String key){
        List<String> keys = new ArrayList<>();
        keys.add(key);

        Long result = stringRedisTemplate.execute(redisScript,keys,"100");
        return result  > 0;
    }
}

 

業務處理

public boolean insert(MsOrder msOrder, User user,Long ms_product_id){
        Assert.notNull(ms_product_id,"購買商品不能為空");

        boolean b = luaReduceStock.reduceStock(RedisConstants.MSSTOCK+ms_product_id);
        if(b){
            //最終搶到庫存的用戶,可以發送一條消息到隊列中,進行非同步下單扣減庫存等。
            Map map = new HashMap();
            map.put("ms_product_id",ms_product_id);
            amqpTemplate.convertAndSend(RabbitConstants.MS_QUEUE,map);
            return true;
        }else{
            serviceError("手慢了,商品已被搶光啦!!!");
        }
        return true;
    }

 

非同步下單,扣減庫存

@Component
@RabbitListener(queues = RabbitConstants.MS_QUEUE)
public class MsOrderHandler {


    @Autowired
    MsProductService msProductService;
    @Resource
    MsProductMapper msProductMapper;
    @Resource
    MsOrderMapper msOrderMapper;

    @RabbitHandler
    public void send(Map map){
        try{
            Long ms_product_id = Long.valueOf(map.get("ms_product_id").toString());
            MsProductDTO msProductDTO = msProductService.findById(ms_product_id);
            MsOrder msOrder = new MsOrder();
            msOrder.setCreated_time(new Date());
            msOrder.setOrder_price(msProductDTO.getMs_price());
            msOrder.setState(1);
            msOrderMapper.insert(msOrder);

            MsProduct msProduct = new MsProduct();
            msProduct.setStock(-1);
            msProduct.setMs_product_id(ms_product_id);
            msProductMapper.updateStock(msProduct);
        }catch (Exception e){
            e.printStackTrace();
        }

    }
}

 

5、jmeter測試

 

 

 

查看執行結果,生成了20條訂單,並且秒殺商品1的庫存減為了0,大功告成!!!

 

 

6、總結

使用Lua腳本調用redis,可以確保操作的原子性,很好地避免了庫存超賣的問題,並且保證了系統的性能,減少網路開銷。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 源碼地址 https://gitee.com/bin-0821/chat-room-demo-go-websocket 關於websocket,上一篇文章講述瞭如何通過websocket進行服務端與客戶端的通信,本篇將會帶領大家把各個websocket進行相互通信,在開始之前,請確保有理解 1 go ...
  • # 流程式控制制練習題 # 一、編程題 1、實現一個課程名稱和課程代號的轉換器:輸入下表中的課程代號,輸出課程的名稱。用戶可以迴圈進行輸入,如果輸入0就退出系統。(**使用****switch +while****迴圈實現**) **課程名稱和課程代號對照表** | **課程名稱** | **課程代碼* ...
  • 看《C++ Primer Plus》時整理的學習筆記,部分內容完全摘抄自《C++ Primer Plus》(第6版)中文版,Stephen Prata 著,張海龍 袁國忠譯。只做學習記錄用途。 ...
  • # 流程式控制制 學習目標: ~~~txt1. idea安裝與使用2. 流程式控制制if...else結構3. 流程式控制制switch結構4. 流程式控制制迴圈結構5. 流程式控制制關鍵字~~~ # 一、流程式控制制概述 什麼是流程式控制制? 流程式控制制是用來控製程序中各語句執行順序的語法。流程式控制制主要包含: * 順序結構 * ...
  • 一、字典、元組的多重嵌套 例 1:記錄全班學生的成績。 分析:定義一個 SimpleGradebook類, 學生名是字典self._grades的鍵,成績是字典self._grades的值。 class SimpleGradebook(): def __init__(self): self._gra ...
  • Django python網路編程回顧 之前我們介紹過web應用程式和http協議,簡單瞭解過web開發的概念。Web應用程式的本質 接收並解析HTTP請求,獲取具體的請求信息 處理本次HTTP請求,即完成本次請求的業務邏輯處理 構造並返回處理結果——HTTP響應 import socket ser ...
  • 三、SpringAMQP SpringAMQP是基於RabbitMQ封裝的一套模板,並且還利用SpringBoot對其實現了自動裝配,使用起來非常方便 SpringAMQP的官方地址 https://spring.io/projects/spring-amqp AMQP Spring AMQP Sp ...
  • 在本篇文章當中主要跟大家介紹併發的基礎知識,從最基本的問題出發層層深入,幫助大家瞭解併發知識,並且打好併發的基礎!!! ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...