MySQL 悲觀鎖、樂觀鎖

閱讀時間約 8 分鐘
假設有一張表,儲存一個iPhone商品跟剩餘庫存量。
假設A/B兩個人同時間進來買商品,如果沒有任何防護機制,資料庫中的商品剩餘庫存量有可能會記錄錯誤,比如iPhone庫存剩10個,此時兩個人"同時"進來搶,由於是同時,所以select到的in memory庫存量都會是10,A/B都會把庫存量-1(10-1)再update資料表,這種情況下剩餘庫存量會是9而不是8,就會發生記錄錯誤的問題。
如果商品庫存剩1個,明明是A先買到,這時候就已經沒庫存了,可是系統卻告訴B恭喜你搶到了,這種情況就更糟糕了,只剩下一支iPhone,卻跟兩個人(甚至更多人)都說有買到,也就是發生所謂的超賣問題。
所以資料庫必須有一些機制來保證資料的正確性,來避免上述所說的問題。

悲觀鎖 Pessimistic Lock

  • 使用資料庫 Transaction 機制來強迫執行順序,確保每次都只有一個事務執行。
每次下SQL時都確保同時只會有一個事務執行,也就是等A SELECT出庫存量並且-1 UPDATE庫存量之後,才釋放出Lock換B執行,這時候B SELECT拿到的庫存數量就會是9,9-1=8再去UPDATE庫存量,也就可以解決數據不正確的問題。
模擬悲觀鎖:
mysql> begin;
mysql> select stock from `product` where id = 1 for update; 
mysql> update `product` set stock = stock -1 where id = 1;
mysql> commit;
可開啟兩個mysql console視窗模擬,A視窗下了for update 會馬上跑出SELECT結果,這時候B視窗也下了這行,會發現沒有馬上執行,必須等待A視窗commit後釋放出lock,B視窗才會繼續執行,等到commit後才又把lock釋放出來。
  • 注意:上述需要支援 Transaction,本文是用InnoDB儲存引擎,MyISAM 是不支援 Transaction 的,如果是用MyISAM會發現for update根本就不會lock住。
  • 悲觀鎖適合寫多讀少的情境。
  • 這種方法的缺點是,如果同時出現大量的讀取操作,由於Lock機制,會需要等待時間,因此可能會有效能問題。

樂觀鎖 Optimistic Lock

  • 因為樂觀,所以DB不會上鎖,但是在update的時候會檢查,在這期間有沒有人更新過資料,一般會加一個version或timestamp欄位來判斷,這種作法並不是利用DB的Transaction,而是一種人為上鎖的機制。
模擬樂觀鎖: 新增一個version欄位
mysql> SELECT stock, version FROM `product` where id = 1;
mysql> update `product` set stock = stock-1, version = version + 1 where id = 1 and version = 0;
兩個視窗同時都會拿到stock=10, version=0的資料,A視窗update資料的時候卡上version=0條件,會更新成功。但這時候B視窗一樣會卡上version=0的條件來更新資料,會發現根本找不到version=0的資料,因為早就被A改成version=1了,也就更新失敗了,這時候應該要通知B使用者購買失敗。

試想如果沒有多version這個欄位當作where條件去更新的話,這筆row data兩個視窗都會更新成功,也就是沒有判斷這段期間有沒有別人更新過,如果庫存剩一個的話,就會發生賣超的問題,剩餘庫存量還記錄為-1。
  • 樂觀鎖適合讀多寫少的情境。
  • 這種方法不適合用於寫多讀少的情況下,因為同時很多併發發生的時候,會有很多update失敗需要重試的狀況發生,消耗的資源開銷會比悲觀鎖更多。

Laravel中使用悲觀鎖(上述的模擬悲觀鎖):

try {
  DB::beginTransaction();
  $data = Product::where('id', 1)->lockForUpdate()->first();
  if($data->stock <= 0){  //庫存不足
    return "庫存不足";
  }
  $data->decrement('stock', 1);
  DB::commit();
}
catch (Exception $e) {
  DB::rollback();
}
transaction簡潔寫法:
$result = DB::transaction(function () {
  $data = Product::where('id', 1)->lockForUpdate()->first();
  if($data->stock <= 0){  //庫存不足
    return false;
  }
  $data->decrement('stock', 1);
  return true;
});
if(!$result){
  return '庫存不足';
}
  • ->lockForUpdate()就是SQL中FOR UPDATE的意思。
  • ->lockForUpdate()要搭配transaction語法一起使用,不然沒有效果。

LOCK IN SHARE MODE

這種鎖法對應到Laravel的->sharedLock()。
比較一下跟FOR UPDATE鎖法差別:
  • LOCK IN SHARE MODE: 不會阻止其它transaction讀取同一行。
  • FOR UPDATE: 會阻止其它transaction讀取同一行(一般沒加鎖的select還是可以讀取,不影響)。
  • 共同之處是兩者都會阻止同一行資料被其它transaction update。
所以如果是超賣的情境,不適合用LOCK IN SHARE MODE,因此使用哪一種lock方法,還是要視系統情境而定,不能亂用。
本筆記參考:
  1. https://www.codeprj.com/zh/blog/bb96e01.html
  2. https://www.gushiciku.cn/pl/gKzx/zh-tw
  3. https://ithelp.ithome.com.tw/articles/10271229
  4. https://www.readfog.com/a/1634624578764509184
  5. https://medium.com/dean-lin/%E7%9C%9F%E6%AD%A3%E7%90%86%E8%A7%A3%E8%B3%87%E6%96%99%E5%BA%AB%E7%9A%84%E6%82%B2%E8%A7%80%E9%8E%96-vs-%E6%A8%82%E8%A7%80%E9%8E%96-2cabb858726d
  6. https://www.w3help.cc/a/202108/569509.html
  7. https://twgreatdaily.com/uTc0AHEBnkjnB-0zi2R_.html
  8. https://blog.51cto.com/u_9443450/2394262
  9. https://ithelp.ithome.com.tw/articles/10267450
  10. https://kknews.cc/zh-tw/news/z9aqnva.html
  11. https://blog.csdn.net/weixin_39947314/article/details/113720708
  12. https://learnku.com/articles/36846
  13. http://www.yangxg.com/blog/id/1530105560
  14. https://learnku.com/articles/57959
  15. https://www.sunzhongwei.com/using-laravel-sharedlock-and-lockforupdate-for-table-row-locks
為什麼會看到廣告
avatar-img
21會員
161內容數
留言0
查看全部
avatar-img
發表第一個留言支持創作者!
Vic Lin的沙龍 的其他內容
Swoole是一個使用C語言寫出來的PHP extension,本篇筆記了如何使用Laravel+Swoole來打造websocket應用,包含從伺服器安裝到基本範例程式,有websocket基本觀念後,再花點時間,就能把程式改成多人聊天室或私頻聊天等應用了。 Requirement 安裝PHP:
Laravel Notifications(通知),是用來通知使用者應用程式訊息的功能,比如付款完成發送email或簡訊通知使用者,文章被訂閱通知等等。Notifications甚至還可以把通知訊息塞進DB,可以用來顯示在後台報表頁面中。 以下以發送email通知來舉例用法。 建立通知 發送通知
Composer是PHP的軟體包管理系統,它提供用於管理PHP軟體和依賴庫關係的標準格式。(引用自維基百科) composer install composer update 這個指令會更新composer.json中指定的套件版本,比如在require中這樣寫: 但如果是這樣: 本筆記參考:
當伺服器需要處理一些比較花時間的任務時(如發送Email、上傳影片等等),讓user等待直到執行完畢,是個很不明智的選擇,這時候就很適合使用Queue,讓工作在背景執行,使用者就能立刻做下一件事,不必在那邊等待。 .env: QUEUE_CONNECTION預設是sync 改成database:
建立middleware指令: 假設建立一個Test middleware: 新增的middleware檔案會在app/Http/Middleware路徑中。 註冊Middleware: Global Middleware: 2. Route Middleware route group用法如下:
Laravel 提供了快速套用軟刪除的方法,直接在Model中加上use SoftDeletes即可: 接著在程式中,假設要把文章1刪除,可以直接這樣寫: 另外,如果我再執行一次上述的Article::find(1)->delete(); 會發現有error: 所以上述改成這樣:
Swoole是一個使用C語言寫出來的PHP extension,本篇筆記了如何使用Laravel+Swoole來打造websocket應用,包含從伺服器安裝到基本範例程式,有websocket基本觀念後,再花點時間,就能把程式改成多人聊天室或私頻聊天等應用了。 Requirement 安裝PHP:
Laravel Notifications(通知),是用來通知使用者應用程式訊息的功能,比如付款完成發送email或簡訊通知使用者,文章被訂閱通知等等。Notifications甚至還可以把通知訊息塞進DB,可以用來顯示在後台報表頁面中。 以下以發送email通知來舉例用法。 建立通知 發送通知
Composer是PHP的軟體包管理系統,它提供用於管理PHP軟體和依賴庫關係的標準格式。(引用自維基百科) composer install composer update 這個指令會更新composer.json中指定的套件版本,比如在require中這樣寫: 但如果是這樣: 本筆記參考:
當伺服器需要處理一些比較花時間的任務時(如發送Email、上傳影片等等),讓user等待直到執行完畢,是個很不明智的選擇,這時候就很適合使用Queue,讓工作在背景執行,使用者就能立刻做下一件事,不必在那邊等待。 .env: QUEUE_CONNECTION預設是sync 改成database:
建立middleware指令: 假設建立一個Test middleware: 新增的middleware檔案會在app/Http/Middleware路徑中。 註冊Middleware: Global Middleware: 2. Route Middleware route group用法如下:
Laravel 提供了快速套用軟刪除的方法,直接在Model中加上use SoftDeletes即可: 接著在程式中,假設要把文章1刪除,可以直接這樣寫: 另外,如果我再執行一次上述的Article::find(1)->delete(); 會發現有error: 所以上述改成這樣:
你可能也想看
Google News 追蹤
Thumbnail
徵的就是你 🫵 超ㄅㄧㄤˋ 獎品搭配超瞎趴的四大主題,等你踹共啦!還有機會獲得經典的「偉士牌樂高」喔!馬上來參加本次的活動吧!
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
本文介紹資料庫鎖(DB lock)的基本概念、鎖的類型及其優缺點。通過對資料庫鎖的深入探討,我們可以理解如何在多用戶環境下保護資料的一致性與完整性,並瞭解排他鎖、共享鎖、意向鎖等不同類型鎖的作用。此外,文章還分析了鎖競爭、鎖等待和死鎖等可能的問題,幫助讀者更全面地掌握資料庫鎖的運作機制。
Thumbnail
※ 為什麼我們需要 Transaction? 當我們談到 Transaction(交易)時,指的是一組不可分割的 SQL 操作。這些操作結果只能成功或失敗,以確保資料庫的一致性和完整性。Transaction 是資料庫操作中的一個「邏輯單位」,包含多個操作步驟。如果其中任何一個步驟失敗,整個 Tr
Thumbnail
已經存在在table裡面的那些record做更新。 ※ 語法 UPDATE [LOW_PRIORITY] [IGNORE] table_name SET column_name1 = expr1, column_name2 = expr2, … [WHERE
Thumbnail
不是你曉得對跟錯,就能夠逃得掉的 - 繁花 別人恐懼我貪婪、別人小賠我破產 先講結論:大盤拉回跳空處停看聽,保守操作保平安 這一週上下千點刷洗,刷新的許多台股成就: 台積電跌停板(鎖死幾秒後打開) 台股史上最大跌幅 台股史上最大漲幅 外資史上最大賣超 外資史上最多空單 自營史上最大
Thumbnail
2024/08/06 [今日交易] 今天終於動手執行了,之前遇過很多次想做的動作, 理論上沒問題,但實際執行就是會遇到一些小問題, 今天是閃電下單夾沒設定好,導致我看不到成交的單子價位在哪, 搞來搞去賺得變小賠,但至少這個執行過的經驗非常重要, 下次應該就沒甚麼問題了, 現貨開盤後所謂
Thumbnail
在競爭激烈的零售行業,庫存管理是成功的關鍵因素之一。有效的庫存管理可以幫助零售商降低成本、提高客戶滿意度並增加銷售。然而,傳統的庫存管理方法面臨諸多挑戰,包括庫存準確性低、成本高和效率低下。
這世界上怎麼會有套牢這件事的發生呢? 只要你對公司營運有信心,你應該要越便宜買的越開心才對啊? 如果真的對公司沒信心,那更應該將資金轉移到更有機會成長的地方才對啊。 這樣也就沒有套牢的事情會發生了,難道你都沒研究就買?(追高)這多半也伴隨著本身沒有選股能力!即時有,(追高的人)也不屑那種成長性
間隔了兩個交易日沒有紀錄,原因有幾個,首先當然是這次又有交易爆炸了,進而影響到心態層面,開始對於自己每天這樣記錄產生懷疑,原因在於自己不斷重複犯一些低級的錯誤,明明每天都記錄下這些錯誤,但老是改不掉,好像這樣的交易日記開始流於形式,並無法真正改善自己的交易狀況,進而產生一種無力感。 這幾天在思考,
Thumbnail
題目敘述 題目會給我們兩張資料表,第一張是Sales,第二張是Product。 第一張是Sales表格,裡面分別有sale_id、 product_id、year、quantity、price等欄位。其中(sale_id、 product_id)做為複合主鍵Primary key Table:
Thumbnail
徵的就是你 🫵 超ㄅㄧㄤˋ 獎品搭配超瞎趴的四大主題,等你踹共啦!還有機會獲得經典的「偉士牌樂高」喔!馬上來參加本次的活動吧!
Thumbnail
隨著理財資訊的普及,越來越多台灣人不再將資產侷限於台股,而是將視野拓展到國際市場。特別是美國市場,其豐富的理財選擇,讓不少人開始思考將資金配置於海外市場的可能性。 然而,要參與美國市場並不只是盲目跟隨標的這麼簡單,而是需要策略和方式,尤其對新手而言,除了選股以外還會遇到語言、開戶流程、Ap
Thumbnail
本文介紹資料庫鎖(DB lock)的基本概念、鎖的類型及其優缺點。通過對資料庫鎖的深入探討,我們可以理解如何在多用戶環境下保護資料的一致性與完整性,並瞭解排他鎖、共享鎖、意向鎖等不同類型鎖的作用。此外,文章還分析了鎖競爭、鎖等待和死鎖等可能的問題,幫助讀者更全面地掌握資料庫鎖的運作機制。
Thumbnail
※ 為什麼我們需要 Transaction? 當我們談到 Transaction(交易)時,指的是一組不可分割的 SQL 操作。這些操作結果只能成功或失敗,以確保資料庫的一致性和完整性。Transaction 是資料庫操作中的一個「邏輯單位」,包含多個操作步驟。如果其中任何一個步驟失敗,整個 Tr
Thumbnail
已經存在在table裡面的那些record做更新。 ※ 語法 UPDATE [LOW_PRIORITY] [IGNORE] table_name SET column_name1 = expr1, column_name2 = expr2, … [WHERE
Thumbnail
不是你曉得對跟錯,就能夠逃得掉的 - 繁花 別人恐懼我貪婪、別人小賠我破產 先講結論:大盤拉回跳空處停看聽,保守操作保平安 這一週上下千點刷洗,刷新的許多台股成就: 台積電跌停板(鎖死幾秒後打開) 台股史上最大跌幅 台股史上最大漲幅 外資史上最大賣超 外資史上最多空單 自營史上最大
Thumbnail
2024/08/06 [今日交易] 今天終於動手執行了,之前遇過很多次想做的動作, 理論上沒問題,但實際執行就是會遇到一些小問題, 今天是閃電下單夾沒設定好,導致我看不到成交的單子價位在哪, 搞來搞去賺得變小賠,但至少這個執行過的經驗非常重要, 下次應該就沒甚麼問題了, 現貨開盤後所謂
Thumbnail
在競爭激烈的零售行業,庫存管理是成功的關鍵因素之一。有效的庫存管理可以幫助零售商降低成本、提高客戶滿意度並增加銷售。然而,傳統的庫存管理方法面臨諸多挑戰,包括庫存準確性低、成本高和效率低下。
這世界上怎麼會有套牢這件事的發生呢? 只要你對公司營運有信心,你應該要越便宜買的越開心才對啊? 如果真的對公司沒信心,那更應該將資金轉移到更有機會成長的地方才對啊。 這樣也就沒有套牢的事情會發生了,難道你都沒研究就買?(追高)這多半也伴隨著本身沒有選股能力!即時有,(追高的人)也不屑那種成長性
間隔了兩個交易日沒有紀錄,原因有幾個,首先當然是這次又有交易爆炸了,進而影響到心態層面,開始對於自己每天這樣記錄產生懷疑,原因在於自己不斷重複犯一些低級的錯誤,明明每天都記錄下這些錯誤,但老是改不掉,好像這樣的交易日記開始流於形式,並無法真正改善自己的交易狀況,進而產生一種無力感。 這幾天在思考,
Thumbnail
題目敘述 題目會給我們兩張資料表,第一張是Sales,第二張是Product。 第一張是Sales表格,裡面分別有sale_id、 product_id、year、quantity、price等欄位。其中(sale_id、 product_id)做為複合主鍵Primary key Table: