Innodb-MVCC詳解

Innodb-MVCC詳解

借用高性能 MySQL 的幾句話

MySQL 的大多數(shù)事務(wù)型存儲引擎都不是簡單的行級鎖,基于提升并發(fā)性能考慮亥曹,他們一般都同時實現(xiàn)了多版本并發(fā)控制 MVCC酌毡,但各自的實現(xiàn)機制不盡相同,因為 MVCC 沒有一個統(tǒng)一的實現(xiàn)標準。

可以認為 MVCC 是行級鎖的一個變種告材,但是他在很多情況下避免了枷鎖操作,因此開銷更低古劲,雖然實現(xiàn)機制有所不同斥赋,但大都實現(xiàn)了非阻塞的讀操作,寫操作也只鎖定必要的行产艾。

我們先來分析為什么個大多數(shù)據(jù)庫都實現(xiàn)了 MVCC 以及實現(xiàn)的好處

  • 項目日常運行中絕大部分的是查詢語句疤剑,如果實現(xiàn)的方式是讀鎖(共享鎖)的方式那么,在大量讀的情況下需要去阻塞寫操作闷堡,如果涉及到了部分寫操作隘膘,大量的鎖爭用問題,可能會導(dǎo)致讀寫操作無法獲得鎖去執(zhí)行杠览。

  • 如果讀不用加鎖弯菊,那么對于系統(tǒng)的吞吐量一定是海量的提升。

  • 寫操作 (insert/update/delete)踱阿,lock 等操作施加最小級別的鎖管钳,依次為主鍵行鎖、唯一索引+主鍵行鎖软舌、間隙鎖蹋嵌、范圍鎖、全表主鍵鎖葫隙。當在其吞吐量較高的情況下栽烂,施加不同的鎖對于系統(tǒng)的性能影響都比較大,需要考驗開發(fā)者的一個技術(shù)水平恋脚。

再來借用高性能 MySQL 的幾句話

  • MVCC 的實現(xiàn)是通過某個時間點的快照來實現(xiàn)的

  • 是通過在每行記錄后面保存系統(tǒng)版本號來實現(xiàn)的腺办,每開啟一個事務(wù),系統(tǒng)版本號遞增并且將值賦值給事務(wù)版本號糟描,然后查詢的時候只會去查找行的版本號早于當前事務(wù)版本號的行...

在這個地方怀喉,因為 它沒有仔細的介紹,相信有的讀者可能會有一些問題船响,如下:

  • 某個時間的快照是什么躬拢,是一個完整的復(fù)制的鏡像文件躲履,還是 mvcc 實現(xiàn),還是數(shù)據(jù)庫數(shù)據(jù)備份等等聊闯?

  • 既然有了當前事務(wù)的快照了工猜,那么讀取的數(shù)據(jù)肯定是當前時間點事務(wù)的快照數(shù)據(jù),為什么還需要 根據(jù)什么系統(tǒng)版本號菱蔬,事務(wù)版本號篷帅,行版本號去 curd 數(shù)據(jù)呢?

通過下來分析你就會明白了

Innodb拴泌,MVCC 的讀可以分為2種魏身,快照讀當前讀蚪腐,這里只介紹 RR 隔離界別

在這之前我們先來了解幾個概念便于我們理解

首先
Innodb 會為每一行數(shù)據(jù)添加3個字段分別是

  • DATA_TRX_ID
    表示當前事務(wù)遞增得到的事務(wù)id作為其行版本號

  • DATA_ROLL_PTR
    一個指向 undo 信息的指針箭昵,undo 就是實現(xiàn) mvcc 的關(guān)鍵,等等會介紹回季。

  • DELETED_BIT
    標識該記錄是否被刪除

其次

我們要注意的一個地方就是家制,快照的創(chuàng)建默認是在你執(zhí)行第一條 SQL 語句的時候,事務(wù)的真正開啟也在這個時候茧跋,我們這里的講解默認開啟事務(wù)就創(chuàng)建快照即執(zhí)行了 start transaction with consistent snapshot;

undo
在 mvcc 中,當用戶讀取一條記錄的時候卓囚,若此記錄已經(jīng)被其它事務(wù)占用瘾杭,當前事務(wù)就可以通過undo log 讀取之前的行的版本信息,找到行版本號小于等于當前事務(wù)版本號的數(shù)據(jù)哪亿。以及讀取到一條被標識為刪除的數(shù)據(jù)的時候粥烁,也可以通過undo log 來獲取之前版本正常的數(shù)據(jù)。這里不理解的話沒有關(guān)系蝇棉,看完下面的內(nèi)容再來看這句話就明白了讨阻。

快照讀

快照數(shù)據(jù)就是當前行數(shù)據(jù)之前的歷史版本,每行記錄都可能有多個版本篡殷,每一行記錄也可能同時存在多個快照數(shù)據(jù)钝吮。所以稱之為 多版本并發(fā)控制 MVCC,每個事務(wù)開啟的時候都會創(chuàng)建一個 read view板辽,它定義在 readOread.h 文件中奇瘦,用來檢索行的可見性

dulint   low_limit_id; // 當前開啟事務(wù)的 id,每開啟一個事務(wù)系統(tǒng)版本號遞增并且賦值給事務(wù)系統(tǒng)版本號
dulint   up_limit_id; // 當前活躍事務(wù)最小的事務(wù) id
ulint    n_trx_ids; // 當前活躍事務(wù) id 的數(shù)量
dulint*  trx_ids; // 當前活躍事務(wù)的 id 列表

每開啟一個事務(wù)的時候和已經(jīng)存在的事務(wù)的記錄的一個記錄

我們假設(shè)現(xiàn)目前用戶 A 開啟了一個事務(wù) id 為 10劲弦,目前存在事務(wù) 9耳标、8、7邑跪、6 即

low_limit_id = 10;
up_limit_id = 6
n_trx_ids = 5
trx_ids = [6,7,8,9,10]

用戶 B 開啟事務(wù)的時候 id 為 9, 存在事務(wù) 8次坡、7呼猪、6

low_limit_id = 9;
up_limit_id = 6
n_trx_ids = 4
trx_ids = [6,7,8,9]

然后我們先分析 select 的幾種情況

  • select
    比如當前處于用戶 B
    只會檢索行的系統(tǒng)版本號小于等于當前事務(wù)的版本號保證數(shù)據(jù)是在事務(wù)開啟之前就已經(jīng)提交了了的或者是自己插入的。
    也就是必須 < 6 (up_limit _id) 的行的版本號才是可見的
    同理對于用戶 A 來說必須是 < 6 的行的系統(tǒng)版本號才是可見的

  • insert/update/delete 屬于當前讀砸琅,下面介紹

當前讀

  • select ... lock in share mode

  • select ... for update

  • insert

  • update

  • delete

當前讀不同于快讀的地方是宋距,當前讀讀取的是系統(tǒng)最新的數(shù)據(jù),它首先回去獲取鎖明棍,獲取到鎖后執(zhí)行對應(yīng)的上述操作乡革,事務(wù)完成后再釋放鎖不是修改完成后。如果這個時間段存在其它的事務(wù)當前讀那么久會阻塞其它事務(wù)摊腋。那么如果這個地方的更新不是主鍵或者唯一索引級別的話會涉及到幻讀沸版,它是采用了間隙鎖(next-key locking)策略防止幻讀的出現(xiàn)。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末兴蒸,一起剝皮案震驚了整個濱河市视粮,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌橙凳,老刑警劉巖蕾殴,帶你破解...
    沈念sama閱讀 216,402評論 6 499
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異岛啸,居然都是意外死亡钓觉,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,377評論 3 392
  • 文/潘曉璐 我一進店門坚踩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來荡灾,“玉大人,你說我怎么就攤上這事瞬铸∨希” “怎么了?”我有些...
    開封第一講書人閱讀 162,483評論 0 353
  • 文/不壞的土叔 我叫張陵嗓节,是天一觀的道長荧缘。 經(jīng)常有香客問我,道長拦宣,這世上最難降的妖魔是什么截粗? 我笑而不...
    開封第一講書人閱讀 58,165評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮鸵隧,結(jié)果婚禮上桐愉,老公的妹妹穿的比我還像新娘。我一直安慰自己掰派,他們只是感情好从诲,可當我...
    茶點故事閱讀 67,176評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般碌奉。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上透敌,一...
    開封第一講書人閱讀 51,146評論 1 297
  • 那天描扯,我揣著相機與錄音定页,去河邊找鬼。 笑死绽诚,一個胖子當著我的面吹牛典徊,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播恩够,決...
    沈念sama閱讀 40,032評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼卒落,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了蜂桶?” 一聲冷哼從身側(cè)響起儡毕,我...
    開封第一講書人閱讀 38,896評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎扑媚,沒想到半個月后腰湾,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,311評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡疆股,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,536評論 2 332
  • 正文 我和宋清朗相戀三年费坊,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片旬痹。...
    茶點故事閱讀 39,696評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡附井,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出唱凯,到底是詐尸還是另有隱情羡忘,我是刑警寧澤谎痢,帶...
    沈念sama閱讀 35,413評論 5 343
  • 正文 年R本政府宣布磕昼,位于F島的核電站,受9級特大地震影響节猿,放射性物質(zhì)發(fā)生泄漏票从。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,008評論 3 325
  • 文/蒙蒙 一滨嘱、第九天 我趴在偏房一處隱蔽的房頂上張望峰鄙。 院中可真熱鬧,春花似錦太雨、人聲如沸吟榴。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽吩翻。三九已至兜看,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間狭瞎,已是汗流浹背细移。 一陣腳步聲響...
    開封第一講書人閱讀 32,815評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留熊锭,地道東北人弧轧。 一個月前我還...
    沈念sama閱讀 47,698評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像碗殷,于是被迫代替她去往敵國和親精绎。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,592評論 2 353

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