溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

MySQL數(shù)據(jù)庫(kù)中有哪些鎖

發(fā)布時(shí)間:2021-08-31 01:28:18 來源:億速云 閱讀:138 作者:chen 欄目:數(shù)據(jù)庫(kù)

這篇文章主要介紹“MySQL數(shù)據(jù)庫(kù)中有哪些鎖”,在日常操作中,相信很多人在MySQL數(shù)據(jù)庫(kù)中有哪些鎖問題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”MySQL數(shù)據(jù)庫(kù)中有哪些鎖”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!

在 MySQL 數(shù)據(jù)庫(kù)中,鎖有很多種類型,不過大致可以分為三類:全局鎖、表級(jí)鎖、行級(jí)鎖。這篇文章我們就簡(jiǎn)單的聊一聊這三種鎖。

全局鎖

全局鎖是粒度比較大的鎖,基本上也使用不上,就像我們家的大門一樣,控制著整個(gè)數(shù)據(jù)庫(kù)實(shí)例。全局鎖就是對(duì)整個(gè)數(shù)據(jù)庫(kù)實(shí)例加鎖,讓整個(gè)數(shù)據(jù)庫(kù)處于只讀狀態(tài)。

MySQL 提供了一個(gè)加全局讀鎖的方法,命令是 Flush tables with read lock  (FTWRL),加鎖之后整個(gè)數(shù)據(jù)庫(kù)實(shí)例處于只讀狀態(tài),有關(guān)數(shù)據(jù)操作的命令都會(huì)被掛起阻塞,例如數(shù)據(jù)更新語(yǔ)句、數(shù)據(jù)定義語(yǔ)句、更新類事務(wù)語(yǔ)句等等。

所以全局鎖一般只用于全庫(kù)備份的時(shí)候,一般只用在不支持一致性讀的存儲(chǔ)引擎做全庫(kù)備份時(shí),比如 MyISAM  這種不支持一致性讀的存儲(chǔ)引擎做全庫(kù)備份時(shí)需要使用全局鎖,像 InnoDB 引擎做全庫(kù)備份時(shí)不需要使用全局鎖。

表級(jí)鎖

表級(jí)鎖是 MySQL 很基本的鎖策略,并且是開銷最小的策略,它鎖住的不是整個(gè)數(shù)據(jù)庫(kù)實(shí)例,而是一張表。

表級(jí)鎖跟全局鎖一樣,MySQL 數(shù)據(jù)庫(kù)提供了加鎖的命令:lock tables … read/write。例如 lock tables t1 read,  t2 write; 命令,則其他線程寫 t1、讀寫 t2 的語(yǔ)句都會(huì)被阻塞。同時(shí),線程 A 在執(zhí)行 unlock tables 之前,也只能執(zhí)行讀 t1、讀寫  t2 的操作。連寫 t1 都不允許,自然也不能訪問其他表。

我們可以使用 unlock tables 主動(dòng)釋放鎖,如果沒有使用的話,在客戶端斷開的時(shí)候自動(dòng)釋放。

表級(jí)鎖存在一個(gè)問題,如果一個(gè)查詢正在遍歷一個(gè)表中的數(shù)據(jù),而執(zhí)行期間另一個(gè)線程對(duì)這個(gè)表結(jié)構(gòu)做變更,刪了一列,那么查詢線程拿到的結(jié)果跟表結(jié)構(gòu)對(duì)不上,肯定是不行的。

為了解決這個(gè)問題,MySQL 5.5版本之后引入了元數(shù)據(jù)鎖(meta data lock,MDL),MDL  是數(shù)據(jù)庫(kù)自動(dòng)加鎖,當(dāng)對(duì)一個(gè)表做增刪改查操作的時(shí)候,加 MDL 讀鎖;當(dāng)要對(duì)表做結(jié)構(gòu)變更操作的時(shí)候,加 MDL 寫鎖。

MDL 鎖有以下兩個(gè)特點(diǎn):

  • 讀鎖之間不互斥,因此你可以有多個(gè)線程同時(shí)對(duì)一張表增刪改查。

  • 讀寫鎖之間、寫鎖之間是互斥的,用來保證變更表結(jié)構(gòu)操作的安全性。因此,如果有兩個(gè)線程要同時(shí)給一個(gè)表加字段,其中一個(gè)要等另一個(gè)執(zhí)行完才能開始執(zhí)行。

行級(jí)鎖

行級(jí)鎖顧名思義就是針對(duì)數(shù)據(jù)庫(kù)表中的行記錄加鎖,行級(jí)鎖可以很大程度的支持并發(fā)處理,但是同時(shí)也帶來了很大的鎖開銷。

行級(jí)鎖比較容易理解,比如事務(wù) A 更新了一行,而這時(shí)候事務(wù) B 也要更新同一行,則必須等事務(wù) A 的操作完成后才能進(jìn)行更新。

行級(jí)鎖是由存儲(chǔ)引擎各自實(shí)現(xiàn)的,也并不是所有的存儲(chǔ)引擎都支持行級(jí)鎖,比如 MyISAM 引擎就不支持行級(jí)鎖,這意味著 MyISAM  存儲(chǔ)引擎要控制并發(fā)只能使用表級(jí)鎖。

InnoDB 引擎實(shí)現(xiàn)了行級(jí)鎖,InnoDB 存儲(chǔ)引擎中實(shí)現(xiàn)了兩種標(biāo)準(zhǔn)的行級(jí)鎖:

  • 共享鎖(S Lock):允許事務(wù)讀一行

  • 排它鎖(X Lock):允許事務(wù)刪除和更新一行

共享鎖是兼容鎖,就是當(dāng)一個(gè)事務(wù)已經(jīng)獲得了行 r 的共享鎖,其他事務(wù)可以立即獲得行 r 的共享鎖,因?yàn)樽x并未改變行 r 的數(shù)據(jù)。

排他鎖是非兼容鎖,如果有事務(wù)想獲取行 r 的排他鎖,若行 r 上有共享鎖或者排它鎖,則它必須等其他事務(wù)釋放行 r 的鎖。

在 InnoDB  存儲(chǔ)引擎中,默認(rèn)情況下使用的是一致性的非鎖定行讀,也就是通過行多版本控制器來讀取行數(shù)據(jù),我們可以顯示的為行加上共享鎖和排它鎖,語(yǔ)句如下:

  • SELECT ..... FOR UPDATE:對(duì)讀取的行記錄加一個(gè)排它鎖,其他事務(wù)想要在這些行上加任何鎖都會(huì)被阻塞

  • SELECT ....... LOCK IN SHARE  MODE:對(duì)讀取的行記錄加一個(gè)共享鎖,其他事務(wù)可以向被鎖定的記錄加共享鎖,但是想要加排它鎖。則會(huì)被阻塞。

到此,關(guān)于“MySQL數(shù)據(jù)庫(kù)中有哪些鎖”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI