色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

深度剖析MySQL/InnoDB的并發控制和加鎖技術

454398 ? 來源:博客園 ? 作者: Yelbosh ? 2020-10-29 14:36 ? 次閱讀

本文主要是針對MySQL/InnoDB的并發控制和加鎖技術做一個比較深入的剖析,并且對其中涉及到的重要的概念,如多版本并發控制(MVCC),臟讀(dirty read),幻讀(phantom read),四種隔離級別(isolation level)等作詳細的闡述,并且基于一個簡單的例子,對MySQL的加鎖進行了一個詳細的分析。本文的總結參考了何登成前輩的博客,并且在前輩總結的基礎上,進行了一些基礎性的說明,希望對剛入門的同學產生些許幫助,如有錯誤,請不吝賜教。按照我的寫作習慣,還是通過幾個關鍵問題來組織行文邏輯,如下:

什么是MVCC(多版本并發控制)?如何理解快照讀(snapshot read)和當前讀(current read)?

什么是隔離級別?臟讀?幻讀?InnoDB的四種隔離級別的含義是什么?

什么是死鎖?

InnoDB是如何實現MVCC的?

一個簡單的sql在不同場景下的加鎖分析

一個復雜的sql的加鎖分析

接下來,我將按照這幾個關鍵問題的順序,對以上問題作一一解答,并且在解答的過程中,爭取將加鎖技術的細節,闡述的更加清楚。

1.1 MVCC:Multi-Version Concurrent Control 多版本并發控制

MVCC是為了實現數據庫的并發控制而設計的一種協議。從我們的直觀理解上來看,要實現數據庫的并發訪問控制,最簡單的做法就是加鎖訪問,即讀的時候不能寫(允許多個西線程同時讀,即共享鎖,S鎖),寫的時候不能讀(一次最多只能有一個線程對同一份數據進行寫操作,即排它鎖,X鎖)。這樣的加鎖訪問,其實并不算是真正的并發,或者說它只能實現并發的讀,因為它最終實現的是讀寫串行化,這樣就大大降低了數據庫的讀寫性能。加鎖訪問其實就是和MVCC相對的LBCC,即基于鎖的并發控制(Lock-Based Concurrent Control),是四種隔離級別中級別最高的Serialize隔離級別。為了提出比LBCC更優越的并發性能方法,MVCC便應運而生。

幾乎所有的RDBMS都支持MVCC。它的最大好處便是,讀不加鎖,讀寫不沖突。在MVCC中,讀操作可以分成兩類,快照讀(Snapshot read)和當前讀(current read)。快照讀,讀取的是記錄的可見版本(可能是歷史版本,即最新的數據可能正在被當前執行的事務并發修改),不會對返回的記錄加鎖;而當前讀,讀取的是記錄的最新版本,并且會對返回的記錄加鎖,保證其他事務不會并發修改這條記錄。在MySQL InnoDB中,簡單的select操作,如select * from table where ? 都屬于快照讀;屬于當前讀的包含以下操作:

select * from table where ? lock in share mode; (加S鎖)

select * from table where ? for update; (加X鎖,下同)

insert, update, delete操作

針對一條當前讀的SQL語句,InnoDB與MySQL Server的交互,是一條一條進行的,因此,加鎖也是一條一條進行的。先對一條滿足條件的記錄加鎖,返回給MySQL Server,做一些DML操作;然后再讀取下一條加鎖,直至讀取完畢。需要注意的是,以上需要加X鎖的都是當前讀,而普通的select(除了for update)都是快照讀,每次insert、update、delete之前都是會進行一次當前讀的,這個時候會上鎖,防止其他事務對某些行數據的修改,從而造成數據的不一致性。我們廣義上說的幻讀現象是通過MVCC解決的,意思是通過MVCC的快照讀可以使得事務返回相同的數據集。如下圖所示:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

注意,我們一般說在MyISAM中使用表鎖,因為MyISAM在修改數據記錄的時候會將整個表鎖起來;而InnoDB使用的是行鎖,即我們以上所談的MVCC的加鎖問題。但是,并不是InnoDB引擎不會使用表鎖,比如在alter table的時候,Innodb就會將該表用表鎖鎖起來。

1.2 隔離級別

在SQL的標準中,定義了四種隔離級別。每一種級別都規定了,在一個事務中所做的修改,哪些在事務內和事務間是可見的,哪些是不可見的。低級別的隔離可以執行更高級別的并發,性能好,但是會出現臟讀和幻讀的現象。首先,我們從兩個基礎的概念說起:

臟讀(dirty read):兩個事務,一個事務讀取到了另一個事務未提交的數據,這便是臟讀。

幻讀(phantom read):兩個事務,事務A與事務B,事務A在自己執行的過程中,執行了兩次相同查詢,第一次查詢事務B未提交,第二次查詢事務B已提交,從而造成兩次查詢結果不一樣,這個其實被稱為不可重復讀;如果事務B是一個會影響查詢結果的insert操作,則好像新多出來的行像幻覺一樣,因此被稱為幻讀。其他事務的提交會影響在同一個事務中的重復查詢結果。

下面簡單描述一下SQL中定義的四種標準隔離級別:

READ UNCOMMITTED (未提交讀):隔離級別:0. 可以讀取未提交的記錄。會出現臟讀。

READ COMMITTED (提交讀):隔離級別:1. 事務中只能看到已提交的修改。不可重復讀,會出現幻讀。(在InnoDB中,會加行所,但是不會加間隙鎖)該隔離級別是大多數數據庫系統的默認隔離級別,但是MySQL的則是RR。

REPEATABLE READ (可重復讀):隔離級別:2.在InnoDB中是這樣的:RR隔離級別保證對讀取到的記錄加鎖 (記錄鎖),同時保證對讀取的范圍加鎖,新的滿足查詢條件的記錄不能夠插入 (間隙鎖),因此不存在幻讀現象。但是標準的RR只能保證在同一事務中多次讀取同樣記錄的結果是一致的,而無法解決幻讀問題。InnoDB的幻讀解決是依靠MVCC的實現機制做到的。

SERIALIZABLE (可串行化):隔離級別:3. 該隔離級別會在讀取的每一行數據上都加上鎖,退化為基于鎖的并發控制,即LBCC。

需要注意的是,MVCC只在RC和RR兩個隔離級別下工作,其他兩個隔離級別都和MVCC不兼容。

1.3 死鎖

死鎖是指兩個或者多個事務在同一資源上相互作用,并請求鎖定對方占用的資源,從而導致惡性循環的現象。當多個事務試圖以不同的順序鎖定資源時,就可能產生死鎖。多個事務同時鎖定同一個資源時,也會產生死鎖。且看下面的兩個產生死鎖的例子:

第一個死鎖很好理解,而第二個死鎖,由于在主索引(聚簇索引表)上仍舊是對兩條記錄進行了不同順序的加鎖,因此仍舊會造成死鎖。死鎖的發生與否,并不在于事務中有多少條SQL語句,死鎖的關鍵在于:兩個(或以上)的Session加鎖的順序不一致。因此,我們通過分析加鎖細節,可以判斷所寫的sql是否會發生死鎖,同時發生死鎖的時候,我們應該如何處理。

1.4 InnoDB的MVCC實現機制

MVCC可以認為是行級鎖的一個變種,它可以在很多情況下避免加鎖操作,因此開銷更低。MVCC的實現大都都實現了非阻塞的讀操作,寫操作也只鎖定必要的行。InnoDB的MVCC實現,是通過保存數據在某個時間點的快照來實現的。一個事務,不管其執行多長時間,其內部看到的數據是一致的。也就是事務在執行的過程中不會相互影響。下面我們簡述一下MVCC在InnoDB中的實現。

InnoDB的MVCC,通過在每行記錄后面保存兩個隱藏的列來實現:一個保存了行的創建時間,一個保存行的過期時間(刪除時間),當然,這里的時間并不是時間戳,而是系統版本號,每開始一個新的事務,系統版本號就會遞增。在RR隔離級別下,MVCC的操作如下:

select操作。a.InnoDB只查找版本早于(包含等于)當前事務版本的數據行。可以確保事務讀取的行,要么是事務開始前就已存在,或者事務自身插入或修改的記錄。b.行的刪除版本要么未定義,要么大于當前事務版本號。可以確保事務讀取的行,在事務開始之前未刪除。

insert操作。將新插入的行保存當前版本號為行版本號。

delete操作。將刪除的行保存當前版本號為刪除標識。

update操作。變為insert和delete操作的組合,insert的行保存當前版本號為行版本號,delete則保存當前版本號到原來的行作為刪除標識。

由于舊數據并不真正的刪除,所以必須對這些數據進行清理,innodb會開啟一個后臺線程執行清理工作,具體的規則是將刪除版本號小于當前系統版本的行刪除,這個過程叫做purge。

1.5 一個簡單SQL的加鎖分析

在MySQL的InnoDB中,都是基于聚簇索引表的。而且普通的select操作都是基于快照讀,是不需要加鎖的。那么我們在分析其他的sql語句的時候,如何分析加鎖細節?下面我們以一個簡單的delete操作的SQL為例,進行一個詳細的闡述。且看下面的SQL:

delete from t1 where id=10;

如果對這條SQL進行加鎖分析,那么MySQL是如何加鎖的呢?一般情況下,我們直觀的感受是:會在id=10的記錄上加鎖。但是,這樣輕率的下結論是片面的,要想確定MySQL的加鎖情況,我們還需要知道更多的條件。還需要知道哪些條件呢?比如:

id列是不是主鍵?

系統的隔離級別是什么?

id非主鍵的話,其上有建立索引嗎?

建立的索引是唯一索引嗎?

該SQL的執行計劃是什么?索引掃描?全表掃描?

接下來,我將這些問題的答案進行組合,然后按照從易到難的順序,逐個分析每種組合下,對應的SQL會加哪些鎖。

組合1:id列是主鍵,RC隔離級別

組合2:id列是二級唯一索引,RC隔離級別

組合3:id列是二級非唯一索引,RC隔離級別

組合4:id列上沒有索引,RC隔離級別

組合5:id列是主鍵,RR隔離級別

組合6:id列是二級唯一索引,RR隔離級別

組合7:id列是二級非唯一索引,RR隔離級別

組合8:id列上沒有索引,RR隔離級別

組合9:Serializable隔離級別

組合1:id列是主鍵,RC隔離級別

當id是主鍵的時候,我們只需要在該id=10的記錄上加上x鎖即可。如下圖所示:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

組合2:id列是二級唯一索引,RC隔離級別

在這里我先解釋一下聚簇索引和普通索引的區別。在InnoDB中,主鍵可以被理解為聚簇索引,聚簇索引中的葉子結點就是相應的數據行,具有聚簇索引的表也被稱為聚簇索引表,數據在存儲的時候,是按照主鍵進行排序存儲的。我們都知道,數據庫在select的時候,會選擇索引列進行查找,索引列都是按照B+樹(多叉搜索樹)數據結構進行存儲,找到主鍵之后,再回到聚簇索引表中進行查詢,這叫回表查詢。那我們自然會問,當使用索引進行查詢的時候,與索引相對應的記錄會被上鎖嗎?會的。如果id是唯一索引,那么只給該唯一索引所對應的索引記錄上x鎖;如果id是非唯一索引,那么所對應的所有的索引記錄上都會上x鎖。如下圖所示:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

組合3:id列是二級非唯一索引,RC隔離級別

解釋同上,如下圖:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

組合4:id列上沒有索引,RC隔離級別

由于id列上沒有索引,因此只能走聚簇索引,進行全部掃描。有人說會在表上加X鎖;有人說會在聚簇索引上,選擇出來的id = 10 的記錄加上X鎖。真實情況如下圖:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

若id列上沒有索引,SQL會走聚簇索引的全掃描進行過濾,由于過濾是由MySQL Server層面進行的。因此每條記錄,無論是否滿足條件,都會被加上X鎖。但是,為了效率考量,MySQL做了優化,對于不滿足條件的記錄,會在判斷后放鎖,最終持有的,是滿足條件的記錄上的鎖,但是不滿足條件的記錄上的加鎖/放鎖動作不會省略。同時,優化也違背了2PL的約束(同時加鎖同時放鎖)。

組合5,6同以上(因為只有一條結果記錄,只能在上面加鎖)

組合7:id列是二級非唯一索引,RR隔離級別

在RR隔離級別下,為了防止幻讀的發生,會使用Gap鎖。這里,你可以把Gap鎖理解為,不允許在數據記錄前面插入數據。首先,通過id索引定位到第一條滿足查詢條件的記錄,加記錄上的X鎖,加GAP上的GAP鎖,然后加主鍵聚簇索引上的記錄X鎖,然后返回;然后讀取下一條,重復進行。直至進行到第一條不滿足條件的記錄[11,f],此時,不需要加記錄X鎖,但是仍舊需要加GAP鎖,最后返回結束。如下圖所示:

組合8:id列無索引,RR隔離級別

在這種情況下,聚簇索引上的所有記錄,都被加上了X鎖。其次,聚簇索引每條記錄間的間隙(GAP),也同時被加上了GAP鎖。如下圖:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

但是,MySQL是做了相關的優化的,就是所謂的semi-consistent read。semi-consistent read開啟的情況下,對于不滿足查詢條件的記錄,MySQL會提前放鎖,同時也不會添加Gap鎖。

組合9:Serializable隔離級別

和RR隔離級別一樣。

1.6 一個復雜的SQL的加鎖分析

這里我們只是列出一個結論,因為要涉及到MySQL的where查詢條件的分析,因此這里先不做詳細介紹,我會在之后的博客中詳細說明。如下圖:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

結論:在RR隔離級別下,針對一個復雜的SQL,首先需要提取其where條件。Index Key確定的范圍,需要加上GAP鎖;Index Filter過濾條件,視MySQL版本是否支持ICP,若支持ICP,則不滿足Index Filter的記錄,不加X鎖,否則需要X鎖;Table Filter過濾條件,無論是否滿足,都需要加X鎖。加鎖的結果如下所示:

深度剖析MySQL/InnoDB的并發控制和加鎖技術

總結

本文只是對MVCC的一些基礎性的知識點進行了詳細的總結,參考了網上和書上比較多的資料和實例。
編輯:hfy

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 死鎖
    +關注

    關注

    0

    文章

    25

    瀏覽量

    8079
  • MySQL
    +關注

    關注

    1

    文章

    817

    瀏覽量

    26623
  • MVCC
    +關注

    關注

    0

    文章

    13

    瀏覽量

    1481
收藏 人收藏

    評論

    相關推薦

    C語言深度剖析

    C語言深度剖析
    發表于 08-25 09:08

    C語言深度剖析

    C語言深度剖析[完整版].pdfC語言深度剖析[完整版].pdf (919.58 KB )
    發表于 03-19 05:11

    詳解Mysql數據庫InnoDB存儲引擎事務

    關于Mysql數據庫InnoDB存儲引擎事務的一點理解
    發表于 05-13 10:11

    由于InnoDB MVCC導致的并發BUG介紹

    [原]記錄一個由于InnoDB MVCC導致的并發BUG
    發表于 07-17 09:46

    InnoDB鎖的特點和狀態查詢

    MySQL探秘(五)InnoDB鎖的類型和狀態查詢
    發表于 08-07 11:45

    分布式MySQLInnoDB cluster

    分布式MySQL——InnoDB cluster和性能測試
    發表于 04-15 08:43

    最有用的mysql問答

    、壓縮、空間函數等,但是不支持事務和行級鎖,所以一般用于有大量查詢少量插入的場景來使用,而且myisam不支持外鍵,并且索引和數據是分開存儲的。 innodb是基于聚簇索引建立的,和myisam相反它支持事務、外鍵,并且通過MVCC來支持高并發,索引和數據存儲在一起。 2
    的頭像 發表于 09-30 17:43 ?1707次閱讀
    最有用的<b class='flag-5'>mysql</b>問答

    MySQL中的高級內容詳解

    MySQL 進階!!! 本文思維導圖如下。 事務控制和鎖定語句 我們知道,MyISAM 和 MEMORY 存儲引擎支持表級鎖定(table-level locking),InnoDB 存儲引擎支持行級鎖定
    的頭像 發表于 03-11 16:55 ?2227次閱讀
    <b class='flag-5'>MySQL</b>中的高級內容詳解

    關于InnoDB的內存結構及原理詳解

    除此之外還聊了一下MySQLInnoDB的日志,和兩次寫,總的來說算是一個入門級別的介紹,這篇文章就來詳細介紹一下InnoDB的內存結構。
    的頭像 發表于 04-16 16:15 ?2797次閱讀
    關于<b class='flag-5'>InnoDB</b>的內存結構及原理詳解

    innodb究竟是如何存數據的

    前言如果你使用過mysql數據庫,對它的存儲引擎:innodb,一定不會感到陌生。 眾所周知,在mysql5以前,默認的存儲引擎是:myslam。但mysql5之后,默認的存儲引擎已經
    的頭像 發表于 10-09 15:41 ?1352次閱讀
    <b class='flag-5'>innodb</b>究竟是如何存數據的

    MySQL5.6 InnoDB支持全文檢索

    在早期的 MySQL 中,InnoDB 并不支持全文檢索技術,從 MySQL 5.6 開始,InnoDB 開始支持全文檢索。
    的頭像 發表于 11-12 15:14 ?1428次閱讀

    剖析MySQL InnoDB存儲原理(上)

    一、MySQL記錄的存儲結構: 1、Page的結構,如下圖:
    的頭像 發表于 02-15 15:45 ?448次閱讀
    <b class='flag-5'>剖析</b><b class='flag-5'>MySQL</b> <b class='flag-5'>InnoDB</b>存儲原理(上)

    剖析MySQL InnoDB存儲原理(下)

    一、InnoDB存儲引擎內存管理 1.1 概念: Buffer Pool:預分配的內存池; Page:Buffer Pool的最小單位; Free list:空閑Page組成的鏈表;
    的頭像 發表于 02-15 15:47 ?421次閱讀
    <b class='flag-5'>剖析</b><b class='flag-5'>MySQL</b> <b class='flag-5'>InnoDB</b>存儲原理(下)

    詳細總結下InnoDB存儲引擎中行鎖的加鎖規則

    對于常見的 DML 語句(如 UPDATE、DELETE 和 INSERT ),InnoDB 會自動給相應的記錄行加寫鎖
    的頭像 發表于 02-21 14:02 ?627次閱讀

    MySQL中的InnoDB是什么?

    有許多強大的MySQL存儲引擎可供我們使用,而InnoDB無疑是最受歡迎的存儲引擎之一。它高度可靠和高效,因此它成為5.5版本以后所有MySQL的默認存儲引擎也就不足為奇了。
    的頭像 發表于 04-13 09:09 ?741次閱讀
    主站蜘蛛池模板: 中文字幕乱偷无码AV蜜桃| 美女脱了内裤张开腿让男人桶到爽| 长篇高h肉爽文丝袜| 影音先锋2017av天堂| 亚洲精品不卡视频| 伊人久久综合热青草| 亚洲人成网站在线播放| 伊人久在线| blacked黑人战小美女| YELLOW视频直播在线观看| 第一福利在线永久视频| 国产老师开裆丝袜喷水漫画| 解开白丝老师的短裙猛烈进入| 曼谷av女郎| 午夜想想爱| 99在线在线视频观看| 国产人人为我我为人人澡| 女人张开腿让男人添| 一本到高清视频在线观看三区| 超碰免费视频caoporn| 久久无码AV亚洲精品色午夜麻豆| 男生jj插入女生jj| 一本大道熟女人妻中文字幕在线| 99re.05久久热最新地址| 国产亚洲综合视频| 欧美性爱 先锋影音| 一起碰一起噜一起草视频| 国产人成精品综合欧美成人| 欧美日韩在线亚洲一| 2021自产拍在线观看视频| 国语对白老女人8av| 无颜之月5集全免费看无删除| 在教室做啊好大用力| 国产亚洲精品AV麻豆狂野| 情浓野战台湾三级| 99久久久无码国产精品免费人妻| 国产亚洲tv在线观看| 亚洲 欧美 清纯 校园 另类| 芳草地在线观看免费视频| 男人插曲视频大全免费网站 | 午夜亚洲国产理论片二级港台二级 |