網站首頁 編程語言 正文
1 什么是秒殺
秒殺最直觀的定義:在高并發場景下而下單某一個商品,這個過程就叫秒殺
【秒殺場景】
- 火車票搶票
- 雙十一限購商品
- 熱度高的明星演唱會門票
- …
2 為什么要防止超賣
早起的12306購票,剛被開發出來使用的時候,12306會經常出現 超賣 這種現象,也就是說車票只剩10張了,卻被20個人買到了,這種現象就是超賣!
還有在高并發的情況下,如果說沒有一定的保護措施,系統會被這種高流量造成宕機
- 庫存100件 你賣了1000件 等著虧錢吧!
- 防止黑客
- 假如我們網站想下發優惠給群眾,但是被黑客利用技術將下發給群眾的利益收入囊中
- 保證用戶體驗
- 高并發場景下,網頁不能打不開、訂單不能支付 要保證網站的使用!
3 單體架構常規秒殺
3.1 常規減庫存代碼
/** * @Author oldlu */ @Service @Transactional //控制事務 public class OrderServiceImpl implements OrderService { @Autowired private StockMapper stockMapper; private OrderMapper orderMapper; //在非并發情況下無問題 @Override public Integer kill(Integer id) { //根據商品id校驗庫存是否還存在 Stock stock = stockMapper.checkStock(id); //當已售和庫存相等就庫存不足了 if(stock.getSale().equals(stock.getCount())){ throw new RuntimeException("庫存不足!"); }else{ //扣除庫存 (已售數量+1) stock.setSale(stock.getSale()+1); stockMapper.updateSale(stock); //更新信息 //創建訂單 Order order = new Order(); order.setSid(stock.getId()).setName(stock.getName()).setCreateDate(new Date()); orderMapper.createOrder(order); //創建訂單 return order.getId(); //mybatis主鍵生成策略 直接返回創建的id } } }
測試controller
/** * @Author oldlu */ @RestController @RequestMapping("/stock") public class StockController { @Autowired private OrderService orderService; //開發秒殺方法 @GetMapping("/kill/{id}") public String kill(@PathVariable("id") Integer id){ System.out.println("秒殺商品的ID=====================>"+id); try { //根據秒殺商品id調用秒殺業務 Integer orderId = orderService.kill(id); return "秒殺成功,訂單ID為:"+String.valueOf(orderId); }catch (Exception e){ e.printStackTrace(); return e.getMessage(); } } }
正常情況看不會有什么問題,就是你訪問一下庫存少一個
3.2 模擬高并發
3.3 超賣現象
3.4 分析原因
線程不安全,方法就是加鎖,單機簡單加鎖即可解決,如果是分布式集群模式搭建那就要考慮分布式鎖
4 簡單實現悲觀樂觀鎖解決單體架構超賣
4.1 悲觀鎖
/** * @Author oldlu */ @RestController @RequestMapping("/stock") public class StockController { @Autowired private OrderService orderService; //開發秒殺方法 @GetMapping("/kill/{id}") public String kill(@PathVariable("id") Integer id){ System.out.println("秒殺商品的ID=====================>"+id); try { //使用悲觀鎖 synchronized (this){ //根據秒殺商品id調用秒殺業務 Integer orderId = orderService.kill(id); return "秒殺成功,訂單ID為:"+String.valueOf(orderId); } }catch (Exception e){ e.printStackTrace(); return e.getMessage(); } } }
這樣效率很差會造成線程阻塞,線程排隊問題,對用戶的體驗不是很好,必須處理完一個才能繼續.
4.2 樂觀鎖
/** * 扣除庫存 * @param stock */ public void updateSale(Stock stock){ //扣除庫存 (已售數量+1) stock.setSale(stock.getSale()+1); stockMapper.updateSale(stock); //更新信息 } /** * 扣除庫存 * @param stock */ public void updateSale(Stock stock){ //在sql層面完成銷量+1 和 版本號 +1 并且根據商品id和版本號同時查詢更新的商品 Integer updRows = stockMapper.updateSale(stock); //更新信息 if(updRows == 0){ //代表沒有拿到版本號 throw new RuntimeException("搶購失敗,請重試!"); } }
也就是沒更新成功說明已經秒殺完了, 相對悲觀鎖而言樂觀鎖保證了一定的效率,而不像悲觀鎖那樣會造成線程阻塞使用樂觀鎖需要使用版本號,在操作數據的時候要對版本號進行更新
4.3 redis鎖setnx
但是上述代碼在高并發,可能其他線程會釋放別人的鎖
4.4 使用Redision
https://github.com/redisson/redisson
5 分布式鎖的解決方案
實現分布式鎖的解決方案
6 采用緩存隊列防止超賣
高并發緩存隊列防止溢出解決方案
原文鏈接:https://blog.csdn.net/ZGL_cyy/article/details/124095922
相關推薦
- 2022-04-12 React工作流程及Error?Boundaries實現過程講解_React
- 2022-12-26 React生命周期與父子組件間通信知識點詳細講解_React
- 2022-12-09 ReactQuery?渲染優化示例詳解_React
- 2022-10-19 react編寫可編輯標題示例詳解_React
- 2022-09-16 Android?運用@JvmName解決函數簽名沖突問題詳解_Android
- 2023-01-26 Kotlin協程Channel源碼示例淺析_Android
- 2023-06-02 關于pip安裝opencv-python遇到的問題_python
- 2022-09-25 Redis時單線程設計的,為什么還這么快
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支