對mysql樂觀鎖、悲觀鎖、共用鎖定、排它鎖、行鎖、表鎖概念的理解

來源:互聯網
上載者:User

標籤:處理   let   sha   相同   分享   資料   exce   where   悲觀鎖   

操作資料庫的時候,可能會由於並發問題而引起的資料的不一致性(資料衝突)

----樂觀鎖

樂觀鎖不是資料庫內建的,需要我們自己去實現。樂觀鎖是指操作資料庫時(更新操作),想法很樂觀,認為這次的操作不會導致衝突,在操作資料時,並不進行任何其他的特殊處理(也就是不加鎖),而在進行更新後,再去判斷是否有衝突了。

通常實現是這樣的:在表中的資料進行操作時(更新),先給資料表加一個版本(version)欄位,每操作一次,將那條記錄的版本號碼加1。也就是先查詢出那條記錄,擷取出version欄位,如果要對那條記錄進行操作(更新),則先判斷此刻version的值是否與剛剛查詢出來時的version的值相等,如果相等,則說明這段期間,沒有其他程式對其進行操作,則可以執行更新,將version欄位的值加1;如果更新時發現此刻的version值與剛剛擷取出來的version的值不相等,則說明這段期間已經有其他程式對其進行操作了,則不進行更新操作。

舉例:

下單操作包括3步驟:

1.查詢出商品資訊

select (status,status,version) from t_goods where id=#{id}

2.根據商品資訊產生訂單

3.修改商品status為2

update t_goods 

set status=2,version=version+1

where id=#{id} and version=#{version};

除了自己手動實現樂觀鎖之外,現在網上許多架構已經封裝好了樂觀鎖的實現,如hibernate,需要時,可能自行搜尋"hiberate 樂觀鎖"試試看。

----悲觀鎖

與樂觀鎖相對應的就是悲觀鎖了。悲觀鎖就是在操作資料時,認為此操作會出現資料衝突,所以在進行每次操作時都要通過擷取鎖才能進行對相同資料的操作,這點跟java中的synchronized很相似,所以悲觀鎖需要耗費較多的時間。另外與樂觀鎖相對應的,悲觀鎖是由資料庫自己實現了的,要用的時候,我們直接調用資料庫的相關語句就可以了。

說到這裡,由悲觀鎖涉及到的另外兩個鎖概念就出來了,它們就是共用鎖定與排它鎖。共用鎖定和排它鎖是悲觀鎖的不同的實現,它倆都屬於悲觀鎖的範疇。

 ----共用鎖定

共用鎖定指的就是對於多個不同的事務,對同一個資源共用同一個鎖。相當於對於同一把門,它擁有多個鑰匙一樣。就像這樣,你家有一個大門,大門的鑰匙有好幾把,你有一把,你女朋友有一把,你們都可能通過這把鑰匙進入你們家,進去啪啪啪啥的,一下理解了哈,沒錯,這個就是所謂的共用鎖定。剛剛說了,對於悲觀鎖,一般資料庫已經實現了,共用鎖定也屬於悲觀鎖的一種,那麼共用鎖定在mysql中是通過什麼命令來調用呢。通過查詢資料,瞭解到通過在執行語句後面加上lock in share mode就代表對某些資源加上共用鎖定了。比如,我這裡通過mysql開啟兩個查詢編輯器,在其中開啟一個事務,並不執行commit語句city表DDL如下:CREATE TABLE `city` (`id` bigint(20) NOT NULL AUTO_INCREMENT,`name` varchar(255) DEFAULT NULL,`state` varchar(255) DEFAULT NULL,PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=18 DEFAULT CHARSET=utf8;

begin;
SELECT * from city where id = "1"  lock in share mode; 然後在另一個查詢時段中,對id為1的資料進行更新  update  city set name="666" where id ="1";此時,操作介面進入了卡頓狀態,過幾秒後,也提示錯誤資訊[SQL]update  city set name="666" where id ="1";
[Err] 1205 - Lock wait timeout exceeded; try restarting transaction 那麼證明,對於id=1的記錄加鎖成功了,在上一條記錄還沒有commit之前,這條id=1的記錄被鎖住了,只有在上一個事務釋放掉鎖後才能進行操作,或用共用鎖定才能對此資料進行操作。再實驗一下:  update city set name="666" where id ="1" lock in share mode;[Err] 1064 - You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ‘lock in share mode‘ at line 1

加上共用鎖定後,也提示錯誤資訊了,通過查詢資料才知道,對於update,insert,delete語句會自動加排它鎖的原因 於是,我又試了試SELECT * from city where id = "1" lock in share mode;
 ok。
  ----排它鎖排它鎖與共用鎖定相對應,就是指對於多個不同的事務,對同一個資源只能有一把鎖。與共用鎖定類型,在需要執行的語句後面加上for update就可以了

 ----行鎖

行鎖,由字面意思理解,就是給某一行加上鎖,也就是一條記錄加上鎖。

比如之前示範的共用鎖定語句

SELECT * from city where id = "1"  lock in share mode; 

由於對於city表中,id欄位為主鍵,就也相當於索引。執行加鎖時,會將id這個索引為1的記錄加上鎖,那麼這個鎖就是行鎖。

 ----表鎖

表鎖:給這個表加上鎖。行鎖:對錶中的行加鎖。

 

對mysql樂觀鎖、悲觀鎖、共用鎖定、排它鎖、行鎖、表鎖概念的理解

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.