搜尋
首頁資料庫Redis解決Redis庫存超賣問題

解決Redis庫存超賣問題

Mar 18, 2021 am 10:49 AM
redis

解決Redis庫存超賣問題

當商品和訂單服務間使用MQ

當商品服務的庫存變更時,透過MQ 通知訂單服務庫存變化。

原始的同步流程

  1. 查詢商品資訊(呼叫商品服務)
  2. 計算總價(產生訂單詳情)
  3. 商品服務扣庫存(呼叫商品服務)
  4. 訂單入庫( 產生訂單)
// 原始的MySQL同步流程// 判断此代金券是否加入抢购SeckillVouchers seckillVouchers = seckillVouchersMapper.selectVoucher(voucherId);AssertUtil.isTrue(seckillVouchers == null, "该代金券并未有抢购活动");// 判断是否有效AssertUtil.isTrue(seckillVouchers.getIsValid() == 0, "该活动已结束");// 插入数据库seckillVouchersMapper.save(seckillVouchers);

#推薦(免費):redis

在訂單產生時直接扣庫存,這是最原始的扣庫存方案,比較簡單,但存在

問題

  • #可能導致許多訂單把產品庫存扣除而未支付,這就需要有一個後台腳本,將一段時間內沒有支付的訂單的庫存釋放,把訂單取消
  • 即時扣庫存,並發差

1、3步驟商品服務,操作商品服務的db,2、4步驟訂單服務,操作訂單服務的db。

避免存取不同服務的 db,原則上同一服務只能操作自身服務的 db。

MQ非同步化

首先考慮只將第4步非同步。

分析

2,4都是操作db,第4步不再等待,1、2、3成功後立即回饋給使用者。

之後透過訊息通知服務非同步下單,若第4步非同步下單失敗,重試操作,試圖重新產生訂單,MQ的訊息也可回溯。

訂單建立完成後,處於排隊狀態,然後服務發布一個事件Order Created 到訊息佇列中。
即訂單服務向外界發送訊息:我建立了一個訂單,由MQ 轉發給訂閱該訊息的服務。

如果商品服務收到建立訂單訊息之後執行扣庫存作業。注意,這裡可能因為某些不可抗因素導致扣庫存失敗,無論成功與否,商品服務都會發送一個扣庫存訊息到 MQ,訊息內容即扣庫存的結果。
訂單服務會訂閱扣庫存的結果,接收到該訊息後:

  • 如果扣庫存成功,將訂單的狀態改為已確認,即下單成功
  • 如果扣庫存失敗,將訂單的狀態改為已取消,即下單失敗

欲實現上述模型要求,需可靠的消息投遞。服務發出的消息,一定會被MQ收到。

  • 使用者體驗的變化
    前端配合排隊中等介面。

商品/訂單服務都變成非同步化,適合秒殺類場景,當流量不大時,並不太適合。

非同步設計

  1. 在庫存在Redis中保存
  2. 收到請求Redis判斷是否有庫存充足,減掉Redis中庫存
  3. 訂單服務建立訂單寫入資料庫,並發送訊息

當訂單支付成功後,會有一個出庫過程,既然有這個過程,就有可能出庫失敗。
庫存有兩個部分:

  • 快取redis層
  • 資料庫mysql層
  1. 當客服新增5個庫存,則快取redis和資料庫mysql層都需增加5個庫存,使用分散式交易的最終一致性來滿足:庫存要麼全加,要麼全不加。
  2. 當訂單產生時,需要扣除庫存,先扣redis庫存,如果扣除成功,則產生訂單進行支付,這個過程不扣除mysql庫存
  3. 當redis庫存扣完,產品就無法下單了,下單就會失敗,就把外層的給擋住了。
  4. 在第2步扣除redis庫存成功後,產生訂單,進行支付,支付成功,返回我的訂單中心, 會發現有一個出庫流程。
  5. 出庫過程
    一個MQ異步解耦的任務佇列,這個過程是扣除mysql庫存
  • 如果扣mysql庫存成功,出庫成功,完成下訂單整個流程,進入發貨狀態
  • 如果扣mysql庫存失敗,出庫失敗,進行一系列的操作
    • #訂單狀態改成取消
    • 返還redis庫存
    • 退款

redis庫存和mysql庫存

支付前是預扣,是扣redis庫存,是鎖定庫存的過程
支付後是真正扣,扣mysql庫存,確保庫存最終一致

但是,在極端情況下會存在資料不一致

  • 如果redis庫存= mysql庫存,不會有問題
  • 如果redis庫存< ; mysql庫存,不會有超賣問題,但會存在實際有庫存,但是沒有賣的情況
  • 如果redis庫存> mysql庫存,就會超賣,超賣的訂單,在出庫的過程中會失敗

這樣整體不會出問題,mysql資料庫層,保證庫存最終不會出問題。

問題

#

資料庫庫存和redis庫存不一致,如何偵測?

如果偵測出來不一致,如何同步

沒有想出來好的方案
比較暴力的方式,就是找一個低高峰期,譬如凌晨1點,週期性強行覆蓋。但極端情況下還是會存在同步後不準確,譬如在同步的過程中,剛好有一個訂單在支付,這個訂單支付成功後,出庫的過程中,扣除了mysql的庫存,但是沒有扣除redis的庫存

這個就是資料庫同步快取的更新機制方面的問題
屬於一致性的邏輯設計的問題
快取數= 資料庫庫存數- 待扣數
當然這裡面也還有其它的方案,以及考慮到一致性的要求高低,可以使用簡單或複雜的方案
就看系統複雜度了,越是大系統就要拆得越細
比如待扣數又可以放到一個佇列裡面,或快取裡面,同時有計數,直接讀計數就行
比如放到mongo,已支付待出庫的數量,一般也不會很大,count一下,也不會損失多少
所以一般系統都無法完全保障資料鏈不出錯,但一定要有補償,就是出錯了可以糾錯
要保障不出錯的代價顯然太大
同步是有一套刷新機制,可以定時,也可以透過MQ,或監控不一至同步等等。 。 。
也叫做保障快取資料的新鮮度
一般不會太長時間,半小時,幾分鐘都有可能,不同場景需求不一樣

12306

買火車票的12306,晚上的時間都不能買票,這個時間估計是在同步庫存,將數據庫庫存同步到redis庫存中, 但是買火車票之類,在訂單生成前,必須扣除實際庫存,也就是要扣除mysql的庫存,

因為買火車票和購物不一樣,購物可以付款後出庫,但是買票這種,支付前就必須出庫,因此,要將出庫流程提前, 只有出庫成功,才能產生訂單,同樣要引入redis庫存

先扣緩存中的庫存,扣除成功後,然後才可以去扣mysql中的庫存。

如果扣除快取中的庫存失敗,就會擋在外面,返回庫存不足,這些請求不會穿刺到mysql中,擋住了大多數的請求壓力。

redis庫存會和mysql庫存不一致,極端情況下是肯定有的,需要進行庫存同步

  • 當快取庫存比資料庫庫存多,那麼就會出現,查詢有票,但是就無法下單,下單的時候就說庫存不足,這個情況下,就會造成數據庫壓力過大,不過12306應該有其他手段來規避這個問題,不過,我確實遇到過,查詢的時候有票,但是無法下單的情況。

  • 當快取庫存比資料庫快取少,那麼不會出問題,只會出現有票,但是沒有出售的情況,等完成庫存同步一下, 明天又準確了。

#

以上是解決Redis庫存超賣問題的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:CSDN。如有侵權,請聯絡admin@php.cn刪除
了解NOSQL:Redis的關鍵特徵了解NOSQL:Redis的關鍵特徵Apr 13, 2025 am 12:17 AM

Redis的關鍵特性包括速度、靈活性和豐富的數據結構支持。 1)速度:Redis作為內存數據庫,讀寫操作幾乎瞬時,適用於緩存和會話管理。 2)靈活性:支持多種數據結構,如字符串、列表、集合等,適用於復雜數據處理。 3)數據結構支持:提供字符串、列表、集合、哈希表等,適合不同業務需求。

REDIS:確定其主要功能REDIS:確定其主要功能Apr 12, 2025 am 12:01 AM

Redis的核心功能是高性能的內存數據存儲和處理系統。 1)高速數據訪問:Redis將數據存儲在內存中,提供微秒級別的讀寫速度。 2)豐富的數據結構:支持字符串、列表、集合等,適應多種應用場景。 3)持久化:通過RDB和AOF方式將數據持久化到磁盤。 4)發布訂閱:可用於消息隊列或實時通信系統。

REDIS:流行數據結構指南REDIS:流行數據結構指南Apr 11, 2025 am 12:04 AM

Redis支持多種數據結構,具體包括:1.字符串(String),適合存儲單一值數據;2.列表(List),適用於隊列和棧;3.集合(Set),用於存儲不重複數據;4.有序集合(SortedSet),適用於排行榜和優先級隊列;5.哈希表(Hash),適合存儲對像或結構化數據。

redis計數器怎麼實現redis計數器怎麼實現Apr 10, 2025 pm 10:21 PM

Redis計數器是一種使用Redis鍵值對存儲來實現計數操作的機制,包含以下步驟:創建計數器鍵、增加計數、減少計數、重置計數和獲取計數。 Redis計數器的優勢包括速度快、高並發、持久性和簡單易用。它可用於用戶訪問計數、實時指標跟踪、遊戲分數和排名以及訂單處理計數等場景。

redis命令行怎麼用redis命令行怎麼用Apr 10, 2025 pm 10:18 PM

使用 Redis 命令行工具 (redis-cli) 可通過以下步驟管理和操作 Redis:連接到服務器,指定地址和端口。使用命令名稱和參數向服務器發送命令。使用 HELP 命令查看特定命令的幫助信息。使用 QUIT 命令退出命令行工具。

redis集群模式怎麼搭建redis集群模式怎麼搭建Apr 10, 2025 pm 10:15 PM

Redis集群模式通過分片將Redis實例部署到多個服務器,提高可擴展性和可用性。搭建步驟如下:創建奇數個Redis實例,端口不同;創建3個sentinel實例,監控Redis實例並進行故障轉移;配置sentinel配置文件,添加監控Redis實例信息和故障轉移設置;配置Redis實例配置文件,啟用集群模式並指定集群信息文件路徑;創建nodes.conf文件,包含各Redis實例的信息;啟動集群,執行create命令創建集群並指定副本數量;登錄集群執行CLUSTER INFO命令驗證集群狀態;使

redis怎麼讀取隊列redis怎麼讀取隊列Apr 10, 2025 pm 10:12 PM

要從 Redis 讀取隊列,需要獲取隊列名稱、使用 LPOP 命令讀取元素,並處理空隊列。具體步驟如下:獲取隊列名稱:以 "queue:" 前綴命名,如 "queue:my-queue"。使用 LPOP 命令:從隊列頭部彈出元素並返回其值,如 LPOP queue:my-queue。處理空隊列:如果隊列為空,LPOP 返回 nil,可先檢查隊列是否存在再讀取元素。

redis集群zset怎麼使用redis集群zset怎麼使用Apr 10, 2025 pm 10:09 PM

Redis 集群中使用 zset:zset 是一種有序集合,將元素與評分關聯。分片策略: a. 哈希分片:根據 zset 鍵的哈希值分佈。 b. 範圍分片:根據元素評分劃分為範圍,並將每個範圍分配給不同的節點。讀寫操作: a. 讀操作:如果 zset 鍵屬於當前節點的分片,則在本地處理;否則,路由到相應的分片。 b. 寫入操作:始終路由到持有 zset 鍵的分片。

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前By尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
4 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

VSCode Windows 64位元 下載

VSCode Windows 64位元 下載

微軟推出的免費、功能強大的一款IDE編輯器

SublimeText3 Linux新版

SublimeText3 Linux新版

SublimeText3 Linux最新版

ZendStudio 13.5.1 Mac

ZendStudio 13.5.1 Mac

強大的PHP整合開發環境

SublimeText3 英文版

SublimeText3 英文版

推薦:為Win版本,支援程式碼提示!

Atom編輯器mac版下載

Atom編輯器mac版下載

最受歡迎的的開源編輯器