面試必備的數(shù)據(jù)庫(kù)悲觀鎖與樂(lè)觀鎖

悲觀鎖與樂(lè)觀鎖

前言

在上一個(gè)章節(jié)5分鐘帶你讀懂事務(wù)隔離性與隔離級(jí)別 的最后映砖,其實(shí)我們已經(jīng)提到了鎖的概念档押。本章節(jié)接下來(lái)將主要介紹以下數(shù)據(jù)庫(kù)悲觀鎖與樂(lè)觀鎖的相關(guān)知識(shí)。如有錯(cuò)誤還請(qǐng)大家及時(shí)指出~

本文已同步至 GitHub/Gitee/公眾號(hào),感興趣的同學(xué)幫忙點(diǎn)波關(guān)注~

問(wèn)題:

  • 為什么需要鎖颜武?
  • 什么是悲觀鎖?
  • 什么是樂(lè)觀鎖拖吼?
  • 悲觀鎖與樂(lè)觀鎖區(qū)別與聯(lián)系鳞上?
  • 悲觀鎖與樂(lè)觀鎖的使用場(chǎng)景?

為什么需要鎖吊档?

在并發(fā)環(huán)境下篙议,如果多個(gè)客戶端訪問(wèn)同一條數(shù)據(jù),此時(shí)就會(huì)產(chǎn)生數(shù)據(jù)不一致的問(wèn)題怠硼,如何解決鬼贱,通過(guò)加鎖的機(jī)制趾断,常見的有兩種鎖,樂(lè)觀鎖和悲觀鎖吩愧,可以在一定程度上解決并發(fā)訪問(wèn)芋酌。

1. 悲觀鎖(Pessimistic Lock)

1.1 定義

百度百科:

悲觀鎖,正如其名雁佳,具有強(qiáng)烈的獨(dú)占和排他特性脐帝。它指的是對(duì)數(shù)據(jù)被外界(包括本系統(tǒng)當(dāng)前的其他事務(wù),以及來(lái)自外部系統(tǒng)的事務(wù)處理)修改持保守態(tài)度糖权,因此堵腹,在整個(gè)數(shù)據(jù)處理過(guò)程中,將數(shù)據(jù)處于鎖定狀態(tài)星澳。悲觀鎖的實(shí)現(xiàn)疚顷,往往依靠數(shù)據(jù)庫(kù)提供的鎖機(jī)制(也只有數(shù)據(jù)庫(kù)層提供的鎖機(jī)制才能真正保證數(shù)據(jù)訪問(wèn)的排他性,否則禁偎,即使在本系統(tǒng)中實(shí)現(xiàn)了加鎖機(jī)制腿堤,也無(wú)法保證外部系統(tǒng)不會(huì)修改數(shù)據(jù))。

其他知識(shí)點(diǎn)

悲觀鎖主要是共享鎖排他鎖
共享鎖又稱為讀鎖如暖,簡(jiǎn)稱S鎖笆檀,顧名思義,共享鎖就是多個(gè)事務(wù)對(duì)于同一數(shù)據(jù)可以共享一把鎖盒至,都能訪問(wèn)到數(shù)據(jù)酗洒,但是只能讀不能修改。
排他鎖又稱為寫鎖枷遂,簡(jiǎn)稱X鎖樱衷,顧名思義,排他鎖就是不能與其他所并存酒唉,如一個(gè)事務(wù)獲取了一個(gè)數(shù)據(jù)行的排他鎖矩桂,其他事務(wù)就不能再獲取該行的其他鎖,包括共享鎖和排他鎖黔州,但是獲取排他鎖的事務(wù)是可以對(duì)數(shù)據(jù)就行讀取和修改耍鬓。

1.2 案例分析

使用場(chǎng)景舉例:以MySQL InnoDB為例

作為演示,我們繼續(xù)使用之前的數(shù)據(jù)庫(kù)表:product表

productId productName productPrice productCount
1 小米 1999 100
2 魅族 1999 100

首先我們需要set autocommit=0流妻,即不允許自動(dòng)提交

有看過(guò)上一篇文章5分鐘帶你讀懂事務(wù)隔離性與隔離級(jí)別 的同學(xué)牲蜀,可以看到最后我們使用事務(wù)隔離級(jí)別時(shí),所引申出來(lái)的根本問(wèn)題就是可以通過(guò)鎖機(jī)制解決绅这。

問(wèn)題

在并發(fā)情況下回導(dǎo)致數(shù)據(jù)一致性的問(wèn)題:
如果有A涣达、B兩個(gè)用戶需要搶productId =1的小米手機(jī),A、B用戶都查詢小米手機(jī)數(shù)量是100度苔,A購(gòu)買后修改商品的數(shù)量為99匆篓,B購(gòu)買后修改數(shù)量為99。

用法

每次獲取小米手機(jī)時(shí)寇窑,對(duì)該商品加排他鎖鸦概。也就是在用戶A獲取獲取 id=1 的小米手機(jī)信息時(shí)對(duì)該行記錄加鎖,期間其他用戶阻塞等待訪問(wèn)該記錄甩骏。代碼如下:

start transaction窗市;
 
select p.productCount from product p where p.productId = 1 for update;
 
update product p set p.productCount=p.productCount-1 where p.productId=1 ;
 
commit;

操作

下面同時(shí)打開兩個(gè)窗口模擬2個(gè)用戶并發(fā)訪問(wèn)數(shù)據(jù)庫(kù)

時(shí)間軸 事務(wù)A 事務(wù)B
T1 start transaction;
T2 select p.productCount from product p where p.productId = 1 for update;
T3 start transaction;
T4 select p.productCount from product p where p.productId = 1 for update;(等待中...)

流程說(shuō)明

  1. 用戶A start transaction開啟一個(gè)事物。前一步我們關(guān)閉了mysql的autocommit饮笛,所以需要手動(dòng)控制事務(wù)的提交咨察。
  2. 在獲得小米手機(jī)信息(productId = 1 )時(shí),進(jìn)行數(shù)據(jù)加鎖操作(for update)福青。與普通查詢方式不同摄狱,我們使用了select…for update的方式,這樣就通過(guò)數(shù)據(jù)庫(kù)實(shí)現(xiàn)了悲觀鎖无午。在這個(gè)update事務(wù)提交之前其他外界是不能修改這條數(shù)據(jù)的媒役,但是這種處理方式效率比較低,一般不推薦使用指厌。
  3. 用戶B start transaction開啟一個(gè)事物刊愚。
  4. 用戶B 也進(jìn)行查詢操作,此時(shí)處于等待中(阻塞狀態(tài))踩验。ps:需要等待用戶A事務(wù)提交后,才會(huì)執(zhí)行商玫。

注意:在事務(wù)中箕憾,只有select…for update(排他鎖) 或lock in share mode(共享鎖) 操作同一個(gè)數(shù)據(jù)時(shí)才會(huì)等待其它事務(wù)結(jié)束后才執(zhí)行,一般select... 則不受此影響拳昌。例如在 T3中執(zhí)行select p.productCount from product p where p.productId = 1;則能正常查詢出數(shù)據(jù)袭异,不會(huì)受第一個(gè)事務(wù)的影響。

2. 樂(lè)觀鎖(Optimistic Lock)

2.1 定義

百度百科:

樂(lè)觀鎖機(jī)制采取了更加寬松的加鎖機(jī)制炬藤。樂(lè)觀鎖是相對(duì)悲觀鎖而言御铃,也是為了避免數(shù)據(jù)庫(kù)幻讀、業(yè)務(wù)處理時(shí)間過(guò)長(zhǎng)等原因引起數(shù)據(jù)處理錯(cuò)誤的一種機(jī)制沈矿,但樂(lè)觀鎖不會(huì)刻意使用數(shù)據(jù)庫(kù)本身的鎖機(jī)制上真,而是依據(jù)數(shù)據(jù)本身來(lái)保證數(shù)據(jù)的正確性。

其他知識(shí)點(diǎn)

實(shí)現(xiàn)樂(lè)觀鎖一般來(lái)說(shuō)有以下2種方式:

  • 使用版本號(hào)
    使用數(shù)據(jù)版本(Version)記錄機(jī)制實(shí)現(xiàn)羹膳,這是樂(lè)觀鎖最常用的一種實(shí)現(xiàn)方式睡互。何謂數(shù)據(jù)版本?即為數(shù)據(jù)增加一個(gè)版本標(biāo)識(shí),一般是通過(guò)為數(shù)據(jù)庫(kù)表增加一個(gè)數(shù)字類型的 “version” 字段來(lái)實(shí)現(xiàn)就珠。當(dāng)讀取數(shù)據(jù)時(shí)寇壳,將version字段的值一同讀出,數(shù)據(jù)每更新一次妻怎,對(duì)此version值加一壳炎。當(dāng)我們提交更新的時(shí)候,判斷數(shù)據(jù)庫(kù)表對(duì)應(yīng)記錄的當(dāng)前版本信息與第一次取出來(lái)的version值進(jìn)行比對(duì)逼侦,如果數(shù)據(jù)庫(kù)表當(dāng)前版本號(hào)與第一次取出來(lái)的version值相等匿辩,則予以更新,否則認(rèn)為是過(guò)期數(shù)據(jù)偿洁。

  • 使用時(shí)間戳
    樂(lè)觀鎖定的第二種實(shí)現(xiàn)方式和第一種差不多撒汉,同樣是在需要樂(lè)觀鎖控制的table中增加一個(gè)字段,名稱無(wú)所謂涕滋,字段類型使用時(shí)間戳(timestamp), 和上面的version類似睬辐,也是在更新提交的時(shí)候檢查當(dāng)前數(shù)據(jù)庫(kù)中數(shù)據(jù)的時(shí)間戳和自己更新前取到的時(shí)間戳進(jìn)行對(duì)比,如果一致則OK宾肺,否則就是版本沖突溯饵。

2.2 案例分析

使用場(chǎng)景舉例:以MySQL InnoDB為例

作為演示,我們繼續(xù)使用之前的數(shù)據(jù)庫(kù)表:product表

productId productName productPrice productCount version
1 小米 1999 100 1
2 魅族 1999 100 2

我們以版本號(hào)實(shí)現(xiàn)的方式進(jìn)行說(shuō)明锨用。

操作

查詢當(dāng)前事務(wù)隔離級(jí)別:


SELECT @@tx_isolation;


結(jié)果:
REPEATABLE-READ

下面同時(shí)打開兩個(gè)窗口模擬2個(gè)用戶并發(fā)訪問(wèn)數(shù)據(jù)庫(kù)

第一種測(cè)試

時(shí)間軸 用戶A 用戶B
T1 start transaction;
T2 select * from product p where p.productId = 1;(productCount=100)
T3 update product p set p.productCount = 99,version=version+1 where p.productId = 1 and version = 1;(受影響的行: 1)
T4 start transaction;
T5 select * from product p where p.productId = 1;(productCount=100)
T6 update product p set p.productCount = 99,version=version+1 where p.productId = 1 and version = 1;(等待中...)
T7 commit;
T8 T6執(zhí)行(受影響的行: 0)
T9 commit;

流程說(shuō)明

  1. 事務(wù)A開啟事務(wù)丰刊。
  2. 事務(wù)A查詢當(dāng)前小米手機(jī)數(shù)量為100。
  3. 事務(wù)A購(gòu)買小米手機(jī)增拥,小米手機(jī)數(shù)量更新為99啄巧。(此時(shí)并未提交事務(wù))。
  4. 事務(wù)B開啟事務(wù)掌栅。
  5. 事務(wù)B查詢當(dāng)前小米手機(jī)數(shù)量為100秩仆。
  6. 事務(wù)B購(gòu)買小米手機(jī),小米手機(jī)數(shù)量更新為99猾封。注意:此時(shí)處于阻塞狀態(tài)澄耍。
  7. 事務(wù)A提交事務(wù)。
  8. 此時(shí)第六步執(zhí)行完畢晌缘,但并未成功(受影響的行: 0)齐莲。
  9. 事務(wù)B提交事務(wù)。

第二種測(cè)試

時(shí)間軸 用戶A 用戶B
T1 select * from product p where p.productId = 1;(productCount=100)
T2 update product p set p.productCount = 99,version=version+1 where p.productId = 1 and version = 1;(受影響的行: 1)
T3 select * from product p where p.productId = 1;(productCount=100)
T4 update product p set p.productCount = 99,version=version+1 where p.productId = 1 and version = 1;(受影響的行: 0)

樂(lè)觀鎖小結(jié)

  • 用戶B修改數(shù)據(jù)的時(shí)候磷箕,受影響行數(shù)為0选酗,對(duì)業(yè)務(wù)來(lái)說(shuō),及更新失敗搀捷。這時(shí)候我們只需要告訴用戶購(gòu)買失敗星掰,重新查詢一遍即可多望。
  • 對(duì)比第一種和第二種測(cè)試,我們會(huì)發(fā)現(xiàn)第一種測(cè)試氢烘,將update語(yǔ)句放入事務(wù)中會(huì)出現(xiàn)阻塞的情況怀偷,而第二種測(cè)試不會(huì)出現(xiàn)阻塞情況。這是為什么呢播玖?update其實(shí)在不在事務(wù)中都無(wú)所謂椎工,在內(nèi)部是這樣的:update是單線程的,及如果一個(gè)線程對(duì)一條數(shù)據(jù)進(jìn)行update操作蜀踏,會(huì)獲得鎖维蒙,其他線程如果要對(duì)同一條數(shù)據(jù)操作會(huì)阻塞,直到這個(gè)線程update成功后釋放鎖果覆。

樂(lè)觀鎖不需要數(shù)據(jù)庫(kù)底層的支持颅痊!

3. 適用場(chǎng)景

悲觀鎖

比較適合寫入操作比較頻繁的場(chǎng)景,如果出現(xiàn)大量的讀取操作局待,每次讀取的時(shí)候都會(huì)進(jìn)行加鎖斑响,這樣會(huì)增加大量的鎖的開銷,降低了系統(tǒng)的吞吐量钳榨。

樂(lè)觀鎖

比較適合讀取操作比較頻繁的場(chǎng)景舰罚,如果出現(xiàn)大量的寫入操作,數(shù)據(jù)發(fā)生沖突的可能性就會(huì)增大薛耻,為了保證數(shù)據(jù)的一致性营罢,應(yīng)用層需要不斷的重新獲取數(shù)據(jù),這樣會(huì)增加大量的查詢操作饼齿,降低了系統(tǒng)的吞吐量饲漾。

文末

本章節(jié)主要簡(jiǎn)單介紹了數(shù)據(jù)庫(kù)中樂(lè)觀鎖與悲觀鎖的相關(guān)知識(shí),后續(xù)我們將會(huì)繼續(xù)介紹數(shù)據(jù)庫(kù)中的其他鎖以及相關(guān)知識(shí)缕溉。例如行鎖能颁、表鎖、死鎖倒淫、

歡迎關(guān)注公眾號(hào):Coder編程
獲取最新原創(chuàng)技術(shù)文章和相關(guān)免費(fèi)學(xué)習(xí)資料,隨時(shí)隨地學(xué)習(xí)技術(shù)知識(shí)败玉!

參考文章:

https://chenzhou123520.iteye.com/blog/1860954

https://chenzhou123520.iteye.com/blog/1863407

微信公眾號(hào)

推薦閱讀

帶你了解數(shù)據(jù)庫(kù)中JOIN的用法

帶你了解數(shù)據(jù)庫(kù)中事務(wù)的ACID特性

5分鐘帶你讀懂事務(wù)隔離性與隔離級(jí)別

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末敌土,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子运翼,更是在濱河造成了極大的恐慌返干,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,743評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件血淌,死亡現(xiàn)場(chǎng)離奇詭異矩欠,居然都是意外死亡财剖,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,296評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門癌淮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)躺坟,“玉大人,你說(shuō)我怎么就攤上這事乳蓄∵涑龋” “怎么了?”我有些...
    開封第一講書人閱讀 157,285評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵虚倒,是天一觀的道長(zhǎng)美侦。 經(jīng)常有香客問(wèn)我,道長(zhǎng)魂奥,這世上最難降的妖魔是什么菠剩? 我笑而不...
    開封第一講書人閱讀 56,485評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮耻煤,結(jié)果婚禮上具壮,老公的妹妹穿的比我還像新娘。我一直安慰自己违霞,他們只是感情好嘴办,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,581評(píng)論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著买鸽,像睡著了一般涧郊。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上眼五,一...
    開封第一講書人閱讀 49,821評(píng)論 1 290
  • 那天妆艘,我揣著相機(jī)與錄音,去河邊找鬼看幼。 笑死批旺,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的诵姜。 我是一名探鬼主播汽煮,決...
    沈念sama閱讀 38,960評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼棚唆!你這毒婦竟也來(lái)了暇赤?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,719評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤宵凌,失蹤者是張志新(化名)和其女友劉穎鞋囊,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體瞎惫,經(jīng)...
    沈念sama閱讀 44,186評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡溜腐,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,516評(píng)論 2 327
  • 正文 我和宋清朗相戀三年译株,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片挺益。...
    茶點(diǎn)故事閱讀 38,650評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡歉糜,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出矩肩,到底是詐尸還是另有隱情现恼,我是刑警寧澤,帶...
    沈念sama閱讀 34,329評(píng)論 4 330
  • 正文 年R本政府宣布黍檩,位于F島的核電站叉袍,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏刽酱。R本人自食惡果不足惜喳逛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,936評(píng)論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望棵里。 院中可真熱鬧润文,春花似錦、人聲如沸殿怜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,757評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)头谜。三九已至骏掀,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間柱告,已是汗流浹背截驮。 一陣腳步聲響...
    開封第一講書人閱讀 31,991評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留际度,地道東北人葵袭。 一個(gè)月前我還...
    沈念sama閱讀 46,370評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像乖菱,于是被迫代替她去往敵國(guó)和親坡锡。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,527評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容