PHP-MALL
1.商城搶購,秒殺庫存超賣是比較頭痛的事,以下使用三種方法防止超賣
1.mysql锁机制,悲观锁InnoDB行级锁方案,不建议使用,对数据库压力较大,如果出现死锁会导致一直不能更新,除非kill掉进程 2.mysql乐观锁 不使用第三方情况下可以使用此方案 3.redis incrby decrby原子性操作,防止超卖 4.为方便扩展,把库存类抽象出接口,方便以后扩展,也可以使用其它方式实现
1.1.mysql鎖定機制,悲觀鎖,InnoDB行級鎖定方案,查詢需使用索引
1.事务级别必须为 SERIALIZABLE 级别 2.查询条件验证库存是否够本次购买,例: id = 1 AND inventory >=1 3.PDO update更新后,不但要验证返回状态是否为!==false,并且同时验证影响行数是否大于0 4.数据库链接一定要使用同一链接,单例或DB链接传入,建议使用单例,由于测试网上找了个db类,没有实现单例,所以使用比较笨的方法,传递数据库链接 5.update件件增加验证购买数量条件 AND inventory >=1
1.2.mysql樂觀鎖定
数据库表增加版本字段如version,每次修改时版本号+1 如果更新操作顺序执行,则数据的版本(version)依次递增,不会产生冲突。但是如果发生有不同的业务操作对同一版本的数据进行修改,那么,先提交的操作(图中B)会把数据version更新为2,当A在B之后提交更新时发现数据的version已经被修改了,那么A的更新操作会失败。 PDO update更新后,不但要验证返回状态是否为true,并且同时验证影响行数是否大于0
1.3.redis原子性操作
incr/decr原子性操作,incr增加,decr减少 //此处不可以取出后放入php变量判断库存,否则会出现幻读,导致超卖 if ($redis_client->decrby($key, $parmas['num']) > -1) { //减库存 $goods = new Goods(); $parmas['version'] = 1; return $goods->subInventory($parmas); } else { //购买多个时,如库存不足,需要把数量加回去,否则会出现库减库存,商品并没有卖出去 $redis_client->incrby($key, $parmas['num']); return false; } 使用redis测试时,每次修改完库存需要删除KEY:DEL goods_1
2.測試
配置使用方式 //db 悲观锁(事务级别SERIALIZABLE) db2 mysql乐观锁 redis redis方式 define('INVENTORY_TYPE', 'db2'); 并发小时可能不会有问题,如查并发较大会有超卖现像,为了可以重现超卖下面代码加入2秒迟时 services/inventory/db/Inventory.php 31行左右 $data = $goods->getInventory($params); sleep(2); //停2秒,方便测试出问题 nginx配置站点mall.com 本机hosts绑定域名 10.211.55.100 mall.com(10.211.55.100为实际站点的ip) 测试方法使用ab测试,请求100,并发10 ab -n100 -c10 http://mall.com
3.問題
注意 由于订单流程中,把验证库存和扣减库存放在了下订单前,可以减少下订单数量和开启的事务数,但如果用户在扣减库存后下订单过程中失败,会出现少卖现象 如果对库存要求不高,可以不用考虑,如果对库存要求较高,需要把扣减成功下单失败对应的订单保存到日志中,然后异步处理恢复库存或给用户补单
以上是商城搶購秒殺防止庫存超賣的詳細內容。更多資訊請關注PHP中文網其他相關文章!
陳述
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

本文討論了PHP中的crypt()和password_hash()的差異,以進行密碼哈希,重點介紹其實施,安全性和對現代Web應用程序的適用性。

文章討論了通過輸入驗證,輸出編碼以及使用OWASP ESAPI和HTML淨化器之類的工具來防止PHP中的跨站點腳本(XSS)。

自動加載PHP會在需要時自動加載類文件,從而通過減少內存使用和增強代碼組織來提高性能。最佳實踐包括使用PSR-4和有效組織代碼。

本文討論了在PHP中管理文件上傳大小的管理,重點是2MB的默認限制以及如何通過修改PHP.INI設置來增加它。

本文討論了PHP 7.1中引入的PHP中的無效類型,允許變量或參數為指定類型或NULL。它突出顯示了諸如提高可讀性,類型安全性和明確意圖的好處,並解釋瞭如何聲明

本文討論了unset()和unlink()功能在編程中的差異,重點關注其目的和用例。 unset()從內存中刪除變量,而unlink()從文件系統中刪除文件。兩者都對效率至關重要


熱AI工具

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

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

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章
Windows 11 KB5054979中的新功能以及如何解決更新問題
3 週前ByDDD
如何修復KB5055523無法在Windows 11中安裝?
2 週前ByDDD
Inzoi:如何申請學校和大學
4 週前ByDDD
如何修復KB5055518無法在Windows 10中安裝?
2 週前ByDDD
Roblox:Dead Rails - 如何召喚和擊敗Nikola Tesla
1 個月前By尊渡假赌尊渡假赌尊渡假赌

熱工具

MantisBT
Mantis是一個易於部署的基於Web的缺陷追蹤工具,用於幫助產品缺陷追蹤。它需要PHP、MySQL和一個Web伺服器。請查看我們的演示和託管服務。

EditPlus 中文破解版
體積小,語法高亮,不支援程式碼提示功能

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

SublimeText3 Linux新版
SublimeText3 Linux最新版

記事本++7.3.1
好用且免費的程式碼編輯器