MySQL悲觀鎖與樂觀鎖的實現方案
前言
悲觀鎖和樂觀鎖是用來解決並發問題的兩種思想,在不同的平臺有著各自的實現。例如在Java中,synchronized就可以認為是悲觀鎖的實現(不嚴謹,有鎖升級的過程,升級到重量級鎖才算),Atomic***原子類可以認為是樂觀鎖的實現。
悲觀鎖
具有強烈的獨占和排他特性,在整個處理過程中將數據處於鎖定狀態,一般是通過系統的互斥量來實現。當其他線程想要獲取鎖時會被阻塞,直到持有鎖的線程釋放鎖。
樂觀鎖
對數據的修改和訪問持樂觀態度,假設不會發生沖突,隻有當數據提交更新時才會對數據沖突與否進行檢測,如果沒有沖突則順利提交更新,否則快速失敗,返回一個錯誤給用戶,讓用戶選擇接下來該如何去做,一般來說失敗後會繼續重試,直到提交更新成功為止。
MySQL本身就支持鎖機制,例如我們有一個「先查再寫」的需求,我們希望整個流程是一個原子操作,中間不能被打斷,這時候就可以通過給查詢的數據行加「排他鎖」來實現。隻要當前事務不釋放鎖,其他事務要想獲得排他鎖,MySQL就會將其阻塞,直到當前事務釋放鎖。這種MySQL底層的排他鎖就稱作「悲觀鎖」。
MySQL本身不提供樂觀鎖的功能,需要開發者自己實現。普遍的做法是在表中加一個version列,用來標記數據行的版本,當我們需要更新數據時,必須比對version版本,version一致說明這個期間數據沒有被其他事務修改過,否則說明數據已經被其他事務修改,需要自旋重試瞭。
實戰
假設數據庫有兩張表:商品表和訂單表。
用戶下單後需要執行兩個操作:
- 商品表減去庫存。
- 訂單表創建一條記錄。
初始數據:ID為1的商品有100的庫存,訂單表數據為空。
客戶端啟動10個線程並發下單,分別在無鎖、悲觀鎖、樂觀鎖的場景下有哪些表現。
如下是創建表的sql語句:
-- 商品表 CREATE TABLE `goods` ( `id` bigint(20) NOT NULL AUTO_INCREMENT, `goods_name` varchar(50) NOT NULL, `price` decimal(10,2) NOT NULL, `stock` int(11) DEFAULT '0', `version` int(10) unsigned NOT NULL DEFAULT '0', PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 -- 訂單表 CREATE TABLE `t_order` ( `id` bigint(20) NOT NULL AUTO_INCREMENT, `goods_id` bigint(20) NOT NULL, `order_time` datetime NOT NULL, PRIMARY KEY (`id`) USING BTREE ) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8
1、無鎖
不做任何處理。
// 下單 private boolean order(){ Goods goods = goodsMapper.selectById(1L); boolean success = false; if (goods.getStock() > 0) { goods.setStock(goods.getStock() - 1); // 更新庫存 goodsMapper.updateById(goods); // 創建訂單 orderMapper.save(goods.getId()); success = true; } return success; }
控制臺輸出結果:
2、悲觀鎖
查詢商品時加FOR UPDATE,給數據行加排他鎖,這樣其他線程再查詢時就會被阻塞,直到當前線程的事務提交並釋放鎖,其他線程才能繼續下單。這種方式並發性能不高。
sql語句
@Select("SELECT * FROM goods WHERE id = #{id} FOR UPDATE") Goods selectForUpdate(Long id);
控制臺輸出結果:
註意:FOR UPDATE必須在事務中才有效,查詢和更新必須在同一個事務中!!!
3、樂觀鎖
實現思路是:每次更新時校驗版本號,如果版本號一致說明期間數據沒有被其他線程改過,當前線程可以正常提交更新,否則說明數據已經被其他線程改過瞭,當前線程需要自旋重試,直到業務成功為止。
更新數據的同時版本號必須自增!!!
@Update("UPDATE goods SET stock = #{stock},version = version+1 WHERE id = #{id} AND version = #{version}") int updateByVersion(Long id, Integer stock, Integer version);
業務代碼
boolean order(){ Goods goods = goodsMapper.selectById(1L); boolean success = false; if (goods.getStock() > 0) { goods.setStock(goods.getStock() - 1); // 更新庫存,帶上版本號 int result = goodsMapper.updateByVersion(goods.getId(), goods.getStock(), goods.getVersion()); if (result <= 0) { // 更新失敗,說明期間數據已經被其他線程修改,需要遞歸重試 return order(); } // 創建訂單 orderMapper.save(goods.getId()); success = true; } return success; }
控制臺輸出結果:
總結
到此這篇關於MySQL悲觀鎖與樂觀鎖方案的文章就介紹到這瞭,更多相關MySQL悲觀鎖與樂觀鎖內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- Redis解決優惠券秒殺應用案例
- mysql自增長id用完瞭該怎麼辦
- springboot配置多數據源的一款框架(dynamic-datasource-spring-boot-starter)
- MySQL深度分頁(千萬級數據量如何快速分頁)
- 為什麼mysql自增主鍵不是連續的