首頁技術(shù)文章正文

MySQL鎖的用法之行級鎖

更新時(shí)間:2018-09-28 來源:黑馬程序員技術(shù)社區(qū) 瀏覽量:

行級鎖是mysql中粒度最小的一種鎖,他能大大減少數(shù)據(jù)庫操作的沖突。但是粒度越小,實(shí)現(xiàn)的成本也越高。myisam引擎只支持表級鎖,而innodb引擎能夠支持行級鎖,下面的內(nèi)容也是針對INNODB行級鎖展開的。

INNODB的行級鎖有共享鎖(S LOCK)和排他鎖(X LOCK)兩種。共享鎖允許事物讀一行記錄,不允許任何線程對該行記錄進(jìn)行修改。排他鎖允許當(dāng)前事物刪除或更新一行記錄,其他線程不能操作該記錄。

  共享鎖:
用法: SELECT ... LOCK IN SHARE MODE;

MySQL會對查詢結(jié)果集中每行都添加共享鎖。

鎖申請前提:當(dāng)前沒有線程對該結(jié)果集中的任何行使用排他鎖,否則申請會阻塞。

操作限制:

使用共享鎖線程與不使用共享鎖線程對鎖定記錄操作限制表

1538118219081_表單.png

1. 使用共享鎖線程可對其鎖定記錄進(jìn)行讀取,其他線程同樣也可對鎖定記錄進(jìn)行讀取操作,并且這兩個(gè)線程讀取的數(shù)據(jù)都屬于同一個(gè)版本。

2. 對于寫入操作,使用共享鎖的線程需要分情況討論,當(dāng)只有當(dāng)前線程對指定記錄使用共享鎖時(shí),線程是可對該記錄進(jìn)行寫入操作(包括更新與刪除),這是由于在寫入操作前,線程向該記錄申請了排他鎖,然后才進(jìn)行寫入操作;當(dāng)其他線程也對該記錄使用共享鎖時(shí),則不可進(jìn)行寫入操作,系統(tǒng)會有報(bào)錯(cuò)提示。不對鎖定記錄使用共享鎖的線程,當(dāng)然是不可進(jìn)行寫入操作了,寫入操作會阻塞。

3. 使用共享鎖進(jìn)程可再次對鎖定記錄申請共享鎖,系統(tǒng)并不報(bào)錯(cuò),但是操作本身并沒有太大意義。其他線程同樣也可以對鎖定記錄申請共享鎖。

4. 使用共享鎖進(jìn)程可對其鎖定記錄申請排他鎖;而其他進(jìn)程是不可以對鎖定記錄申請排他鎖,申請會阻塞。
  排他鎖:

用法: SELECT ... FOR UPDATE;

MySQL會對查詢結(jié)果集中每行都添加排他鎖,在事物操作中,任何對記錄的更新與刪除操作會自動(dòng)加上排他鎖。

    鎖申請前提:當(dāng)前沒有線程對該結(jié)果集中的任何行使用排他鎖或共享鎖,否則申請會阻塞。

操作限制:

使用排他鎖線程與不使用排他鎖線程對鎖定記錄操作限制表

線程    讀取操作    寫入操作    共享鎖申請    排他鎖申請    

使用排他鎖    可讀(新版本)    可寫    可申請    可申請    

不使用排他鎖    可讀(舊版本)    不可寫(阻塞)    不可申請(阻塞)    不可申請(阻塞)    


1. 使用排他鎖線程可以對其鎖定記錄進(jìn)行讀取,讀取的內(nèi)容為當(dāng)前事物的最新版本;而對于不使用排他鎖的線程,同樣是可以進(jìn)行讀取操作,這種特性是一致性非鎖定讀。即對于同一條記錄,數(shù)據(jù)庫記錄多個(gè)版本,在事物內(nèi)的更新操作會反映到新版本中,而舊版本會提供給其他線程進(jìn)行讀取操作。

2. 使用排他鎖線程可對其鎖定記錄進(jìn)行寫入操作;對于不使用排他鎖的線程,對鎖定記錄的寫操作是不允許的,請求會阻塞。

3. 使用排他鎖進(jìn)程可對其鎖定記錄申請共享鎖,但是申請共享鎖之后,線程并不會釋放原先的排他鎖,因此該記錄對外表現(xiàn)出排他鎖的性質(zhì);其他線程是不可對已鎖定記錄申請共享鎖,請求會阻塞。

4. 使用排他鎖進(jìn)程可對其鎖定記錄申請排他鎖(實(shí)際上并沒有任何意義);而其他進(jìn)程是不可對鎖定記錄申請排他鎖,申請會阻塞。

   

本文版權(quán)歸黑馬程序員JavaEE學(xué)院所有,歡迎轉(zhuǎn)載,轉(zhuǎn)載請注明作者出處。謝謝!

作者:黑馬程序員PHP+H5全棧培訓(xùn)學(xué)院

首發(fā):http://php.itheima.com/?v2

分享到:
在線咨詢 我要報(bào)名
和我們在線交談!