高併發mysql update操作必定失敗
========
$database = new medoo(Database::connectdb());
//這個表草雞簡單啊,update
$database->update('visits', array(
'total[+]' => 1
));
*/
========++++++++++++
'UPDATE "' . $table . '" SET ' ==== XXXXXXX
是不是可以用redis
========++++++++++++
今天王總又給我們上了一課,其實MySQL處理高併發,防止庫存超賣的問題,在去年的時候,王總已經提過;但是很可惜,即使當時大家都聽懂了,但是在現實開發中,還是沒這方面的意識。今天就我的一些理解,整理一下這個問題,並希望以後這樣的課程能多點。
先來就庫存超賣的問題作描述:一般電子商務網站都會遇到如團購、秒殺、特價之類的活動,而這樣的活動有一個共同的特點就是訪問量激增、上千甚至上萬人搶購一個商品。然而,作為活動商品,庫存肯定是很有限的,如何控制庫存不讓出現超買,以防止造成不必要的損失是眾多電子商務網站程式設計師頭疼的問題,這同時也是最基本的問題。
從技術方面剖析,很多人肯定會想到事務,但是事務是控制庫存超賣的必要條件,但不是充分必要條件。
舉例:
總庫存:4個商品
請求人:a、1個商品 b、2個商品 c、3個商品
程式如下:
beginTranse(開啟事務)
try{
$result = $dbca->query('select amount from s_store where postID = 12345');
if(result->amount > 0){
//quantity為請求減掉的庫存數量
$dbca->query('update s_store set amount = amount - quantity where postID = 12345');
}
}catch($e Exception){
rollBack(回滾)
}
commit(提交事務)
以上程式碼就是我們平時控制庫存寫的程式碼了,大多數人都會這麼寫,看似問題不大,其實隱藏著巨大的漏洞。資料庫的訪問其實就是對磁碟檔案的訪問,資料庫中的表其實就是儲存在磁碟上的一個個檔案,甚至一個檔案包含了多張表。例如由於高併發,當前有三個使用者a、b、c三個使用者進入到了這個事務中,這個時候會產生一個共享鎖,所以在select的時候,這三個使用者查到的庫存數量都是4個,同時還要注意,mysql innodb查到的結果是有版本控制的,再其他使用者更新沒有commit之前(也就是沒有產生新版本之前),當前使用者查到的結果依然是就版本;
然後是update,假如這三個使用者同時到達update這裡,這個時候update更新語句會把併發序列化,也就是給同時到達這裡的是三個使用者排個序,一個一個執行,並生成排他鎖,在當前這個update語句commit之前,其他使用者等待執行,commit後,生成新的版本;這樣執行完後,庫存肯定為負數了。但是根據以上描述,我們修改一下程式碼就不會出現超買現象了,程式碼如下:
beginTranse(開啟事務)
try{
//quantity為請求減掉的庫存數量
$dbca->query('update s_store set amount = amount - quantity where postID = 12345');
$result = $dbca->query('select amount from s_store where postID = 12345');
if(result->amount < 0){
throw new Exception('庫存不足');
}
}catch($e Exception){
rollBack(回滾)
}
commit(提交事務)
另外,更簡潔的方法:
beginTranse(開啟事務)
try{
//quantity為請求減掉的庫存數量
$dbca->query('update s_store set amount = amount - quantity where amount>=quantity and postID = 12345');
}catch($e Exception){
rollBack(回滾)
}
commit(提交事務)
=====================================================================================
1、在秒殺的情況下,肯定不能如此高頻率的去讀寫資料庫,會嚴重造成效能問題的
必須使用快取,將需要秒殺的商品放入快取中,並使用鎖來處理其併發情況。當接到使用者秒殺提交訂單的情況下,先將商品數量遞減(加鎖/解鎖)後再進行其他方面的處理,處理失敗在將資料遞增1(加鎖/解鎖),否則表示交易成功。
當商品數量遞減到0時,表示商品秒殺完畢,拒絕其他使用者的請求。
2、這個肯定不能直接運算元據庫的,會掛的。直接讀庫寫庫對資料庫壓力太大,要用快取。
把你要賣出的商品比如10個商品放到快取中;然後在memcache裡設定一個計數器來記錄請求數,這個請求書你可以以你要秒殺賣出的商品數為基數,比如你想賣出10個商品,只允許100個請求進來。那當計數器達到100的時候,後面進來的就顯示秒殺結束,這樣可以減輕你的伺服器的壓力。然後根據這100個請求,先付款的先得後付款的提示商品以秒殺完。
3、首先,多使用者併發修改同一條記錄時,肯定是後提交的使用者將覆蓋掉前者提交的結果了。
這個直接可以使用加鎖機制去解決,樂觀鎖或者悲觀鎖。
樂觀鎖,就是在資料庫設計一個版本號的欄位,每次修改都使其+1,這樣在提交時比對提交前的版本號就知道是不是併發提交了,但是有個缺點就是隻能是應用中控制,如果有跨應用修改同一條資料樂觀鎖就沒辦法了,這個時候可以考慮悲觀鎖。
悲觀鎖,就是直接在資料庫層面將資料鎖死,類似於oralce中使用select xxxxx from xxxx where xx=xx for update,這樣其他執行緒將無法提交資料。
除了加鎖的方式也可以使用接收鎖定的方式,思路是在資料庫中設計一個狀態標識位,使用者在對資料進行修改前,將狀態標識位標識為正在編輯的狀態,這樣其他使用者要編輯此條記錄時系統將發現有其他使用者正在編輯,則拒絕其編輯的請求,類似於你在作業系統中某檔案正在執行,然後你要修改該檔案時,系統會提醒你該檔案不可編輯或刪除。
4、不建議在資料庫層面加鎖,建議通過服務端的記憶體鎖(鎖主鍵)。當某個使用者要修改某個id的資料時,把要修改的id存入memcache,若其他使用者觸發修改此id的資料時,讀到memcache有這個id的值時,就阻止那個使用者修改。
5、實際應用中,並不是讓mysql去直面大併發讀寫,會藉助“外力”,比如快取、利用主從庫實現讀寫分離、分表、使用佇列寫入等方法來降低併發讀寫。
相關文章
- 【MySQL】ERROR 1175 安全模式UPDATE/DELETE操作失敗MySqlError模式delete
- mysql 高併發 select update 併發更新問題解決方案MySql
- GreatSQL執行Update失敗案例分析SQL
- mysql併發操作問題MySql
- MySQL 常用的UPDATE操作MySql
- MySQL中SELECT+UPDATE併發更新問題MySql
- win10 windows update更新失敗怎麼解決 win10系統update更新失敗如何修復Win10Windows
- rosdep update 超時失敗最新解決方法ROS
- windows update更新失敗報錯解決方法Windows
- win8.1 update更新失敗解決方法
- MySQL啟動失敗MySql
- Mysql update誤操作恢復MySql
- MySQL 高併發配置最佳化MySql
- win8.1update安裝失敗怎麼辦?
- ubuntu 12.10 sudo apt-get update 失敗Ubuntuapt-get
- 聊聊sql的併發updateSQL
- 高併發下丟失更新的解決方案
- Java使用程式碼模擬高併發操作Java
- PHP利用Mysql鎖解決高併發PHPMySql
- mysql大資料高併發處理MySql大資料
- [分散式][高併發]高併發架構分散式架構
- Mysql備份失敗案例(一)MySql
- Kali 系統 apt-get update更新失敗解決方案apt-get
- MySQL MyISAM/InnoDB高併發優化經驗MySql優化
- 發郵件失敗,求助
- 安裝CRS失敗後的清除操作
- 一次失敗的logmnr操作
- 操作都失敗,是什麼原因啊
- java高併發系列 - 第21天:java中的CAS操作,java併發的基石Java
- MySQL建立表失敗的問題MySql
- mysql操作命令梳理(2)-alter(update、insert)MySql
- MySQL Study之–Mysql啟動失敗“mysql.host”薦MySql
- mysql 高併發檔案系統JFS安裝MySql
- MySQL資料庫高併發最佳化配置MySql資料庫
- 構建高併發&高可用&安全的IT系統-高併發部分
- OpenResty高併發REST
- 高併發(鎖)
- OpenSIPS 2.4.2 高併發下,日誌丟失怎麼辦