這篇文章將為大家詳細(xì)講解有關(guān)MySQL中事務(wù)隔離級(jí)別的實(shí)現(xiàn)原理是什么,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對(duì)相關(guān)知識(shí)有一定的了解。
創(chuàng)新互聯(lián)堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的呼和浩特網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!隔離性(isolation)指的是不同事務(wù)先后提交并執(zhí)行后,最終呈現(xiàn)出來的效果是串行的,也就是說,對(duì)于事務(wù)來說,它在執(zhí)行過程中,感知到的數(shù)據(jù)變化應(yīng)該只有自己操作引起的,不存在其他事務(wù)引發(fā)的數(shù)據(jù)變化。
隔離性解決的是并發(fā)事務(wù)出現(xiàn)的問題。
隔離性最簡(jiǎn)單的實(shí)現(xiàn)方式就是各個(gè)事務(wù)都串行執(zhí)行了,如果前面的事務(wù)還沒有執(zhí)行完畢,后面的事務(wù)就都等待。但是這樣的實(shí)現(xiàn)方式很明顯并發(fā)效率不高,并不適合在實(shí)際環(huán)境中使用。
為了解決上述問題,實(shí)現(xiàn)不同程度的并發(fā)控制,SQL的標(biāo)準(zhǔn)制定者提出了不同的隔離級(jí)別:未提交讀(read uncommitted)、提交讀(read committed)、可重復(fù)讀(repeatable read)、序列化讀(serializable)。其中高級(jí)隔離級(jí)別就是序列化讀,而在其他隔離級(jí)別中,由于事務(wù)是并發(fā)執(zhí)行的,所以或多或少允許出現(xiàn)一些問題。見以下的矩陣表:
隔離級(jí)別(+:允許出現(xiàn),-:不允許出現(xiàn)) | 臟讀 | 不可重復(fù)讀 | 幻讀 |
---|---|---|---|
未提交讀 | + | + | + |
提交讀 | - | + | + |
可重復(fù)讀 | - | - | + |
序列化讀 | - | - | - |
注意,MySQL的InnoDB引擎在提交讀級(jí)別通過MVCC解決了不可重復(fù)讀的問題,在可重復(fù)讀級(jí)別通過間隙鎖解決了幻讀問題,具體見下面的分析。
我們上面遇到的問題其實(shí)就是并發(fā)事務(wù)下的控制問題,解決并發(fā)事務(wù)的最常見方式就是悲觀并發(fā)控制了(也就是數(shù)據(jù)庫中的鎖)。標(biāo)準(zhǔn)SQL事務(wù)隔離級(jí)別的實(shí)現(xiàn)是依賴鎖的,我們來看下具體是怎么實(shí)現(xiàn)的:
事務(wù)隔離級(jí)別 | 實(shí)現(xiàn)方式 |
---|---|
未提交讀(RU) | 事務(wù)對(duì)當(dāng)前被讀取的數(shù)據(jù)不加鎖; 事務(wù)在更新某數(shù)據(jù)的瞬間(就是發(fā)生更新的瞬間),必須先對(duì)其加行級(jí)共享鎖,直到事務(wù)結(jié)束才釋放。 |
提交讀(RC) | 事務(wù)對(duì)當(dāng)前被讀取的數(shù)據(jù)加行級(jí)共享鎖(當(dāng)讀到時(shí)才加鎖),一旦讀完該行,立即釋放該行級(jí)共享鎖; 事務(wù)在更新某數(shù)據(jù)的瞬間(就是發(fā)生更新的瞬間),必須先對(duì)其加行級(jí)排他鎖,直到事務(wù)結(jié)束才釋放。 |
可重復(fù)讀(RR) | 事務(wù)在讀取某數(shù)據(jù)的瞬間(就是開始讀取的瞬間),必須先對(duì)其加行級(jí)共享鎖,直到事務(wù)結(jié)束才釋放; 事務(wù)在更新某數(shù)據(jù)的瞬間(就是發(fā)生更新的瞬間),必須先對(duì)其加行級(jí)排他鎖,直到事務(wù)結(jié)束才釋放。 |
序列化讀(S) | 事務(wù)在讀取數(shù)據(jù)時(shí),必須先對(duì)其加表級(jí)共享鎖 ,直到事務(wù)結(jié)束才釋放; 事務(wù)在更新數(shù)據(jù)時(shí),必須先對(duì)其加表級(jí)排他鎖 ,直到事務(wù)結(jié)束才釋放。 |
可以看到,在只使用鎖來實(shí)現(xiàn)隔離級(jí)別的控制的時(shí)候,需要頻繁的加鎖解鎖,而且很容易發(fā)生讀寫的沖突(例如在RC級(jí)別下,事務(wù)A更新了數(shù)據(jù)行1,事務(wù)B則在事務(wù)A提交前讀取數(shù)據(jù)行1都要等待事務(wù)A提交并釋放鎖)。
為了不加鎖解決讀寫沖突的問題,MySQL引入了MVCC機(jī)制,詳細(xì)可見我以前的分析文章:一文讀懂?dāng)?shù)據(jù)庫中的樂觀鎖和悲觀鎖和MVCC。
在往下分析之前,我們有幾個(gè)概念需要先了解下:
1、鎖定讀和一致性非鎖定讀
鎖定讀:在一個(gè)事務(wù)中,主動(dòng)給讀加鎖,如SELECT ... LOCK IN SHARE MODE 和 SELECT ... FOR UPDATE。分別加上了行共享鎖和行排他鎖。鎖的分類可見我以前的分析文章:你應(yīng)該了解的MySQL鎖分類)。
https://dev.mysql.com/doc/refman/8.0/en/innodb-locking-reads.html
一致性非鎖定讀:InnoDB使用MVCC向事務(wù)的查詢提供某個(gè)時(shí)間點(diǎn)的數(shù)據(jù)庫快照。查詢會(huì)看到在該時(shí)間點(diǎn)之前提交的事務(wù)所做的更改,而不會(huì)看到稍后或未提交的事務(wù)所做的更改(本事務(wù)除外)。也就是說在開始了事務(wù)之后,事務(wù)看到的數(shù)據(jù)就都是事務(wù)開啟那一刻的數(shù)據(jù)了,其他事務(wù)的后續(xù)修改不會(huì)在本次事務(wù)中可見。
Consistent read是InnoDB在RC和RR隔離級(jí)別處理SELECT語句的默認(rèn)模式。一致性非鎖定讀不會(huì)對(duì)其訪問的表設(shè)置任何鎖,因此,在對(duì)表執(zhí)行一致性非鎖定讀的同時(shí),其它事務(wù)可以同時(shí)并發(fā)的讀取或者修改它們。
https://dev.mysql.com/doc/refman/8.0/en/innodb-consistent-read.html
2、當(dāng)前讀和快照讀
當(dāng)前讀
讀取的是新版本,像UPDATE、DELETE、INSERT、SELECT ... LOCK IN SHARE MODE、SELECT ... FOR UPDATE這些操作都是一種當(dāng)前讀,為什么叫當(dāng)前讀?就是它讀取的是記錄的新版本,讀取時(shí)還要保證其他并發(fā)事務(wù)不能修改當(dāng)前記錄,會(huì)對(duì)讀取的記錄進(jìn)行加鎖。
快照讀
讀取的是快照版本,也就是歷史版本,像不加鎖的SELECT操作就是快照讀,即不加鎖的非阻塞讀;快照讀的前提是隔離級(jí)別不是未提交讀和序列化讀級(jí)別,因?yàn)槲刺峤蛔x總是讀取新的數(shù)據(jù)行,而不是符合當(dāng)前事務(wù)版本的數(shù)據(jù)行,而序列化讀則會(huì)對(duì)表加鎖。
3、隱式鎖定和顯式鎖定
隱式鎖定
InnoDB在事務(wù)執(zhí)行過程中,使用兩階段鎖協(xié)議(不主動(dòng)進(jìn)行顯示鎖定的情況):
隨時(shí)都可以執(zhí)行鎖定,InnoDB會(huì)根據(jù)隔離級(jí)別在需要的時(shí)候自動(dòng)加鎖;
鎖只有在執(zhí)行commit或者rollback的時(shí)候才會(huì)釋放,并且所有的鎖都是在同一時(shí)刻被釋放。
顯式鎖定
InnoDB也支持通過特定的語句進(jìn)行顯示鎖定(存儲(chǔ)引擎層)
select ... lock in share mode //共享鎖 select ... for update //排他鎖
MySQL Server層的顯示鎖定:
lock table unlock table
了解完上面的概念后,我們來看下InnoDB的事務(wù)具體是怎么實(shí)現(xiàn)的(下面的讀都指的是非主動(dòng)加鎖的select)
事務(wù)隔離級(jí)別 | 實(shí)現(xiàn)方式 |
---|---|
未提交讀(RU) | 事務(wù)對(duì)當(dāng)前被讀取的數(shù)據(jù)不加鎖,都是當(dāng)前讀; 事務(wù)在更新某數(shù)據(jù)的瞬間(就是發(fā)生更新的瞬間),必須先對(duì)其加行級(jí)共享鎖,直到事務(wù)結(jié)束才釋放。 |
提交讀(RC) | 事務(wù)對(duì)當(dāng)前被讀取的數(shù)據(jù)不加鎖,且是快照讀; 事務(wù)在更新某數(shù)據(jù)的瞬間(就是發(fā)生更新的瞬間),必須先對(duì)其加行級(jí)排他鎖(Record),直到事務(wù)結(jié)束才釋放。 通過快照,在這個(gè)級(jí)別MySQL就解決了不可重復(fù)讀的問題 |
可重復(fù)讀(RR) | 事務(wù)對(duì)當(dāng)前被讀取的數(shù)據(jù)不加鎖,且是快照讀; 事務(wù)在更新某數(shù)據(jù)的瞬間(就是發(fā)生更新的瞬間),必須先對(duì)其加行級(jí)排他鎖(Record,GAP,Next-Key),直到事務(wù)結(jié)束才釋放。 通過間隙鎖,在這個(gè)級(jí)別MySQL就解決了幻讀的問題 |
序列化讀(S) | 事務(wù)在讀取數(shù)據(jù)時(shí),必須先對(duì)其加表級(jí)共享鎖 ,直到事務(wù)結(jié)束才釋放,都是當(dāng)前讀; 事務(wù)在更新數(shù)據(jù)時(shí),必須先對(duì)其加表級(jí)排他鎖 ,直到事務(wù)結(jié)束才釋放。 |
可以看到,InnoDB通過MVCC很好的解決了讀寫沖突的問題,而且提前一個(gè)級(jí)別就解決了標(biāo)準(zhǔn)級(jí)別下會(huì)出現(xiàn)的幻讀和不可重復(fù)讀問題,大大提升了數(shù)據(jù)庫的并發(fā)能力。
不可重復(fù)讀:前后多次讀取一行,數(shù)據(jù)內(nèi)容不一致,針對(duì)其他事務(wù)的update和delete操作。為了解決這個(gè)問題,使用行共享鎖,鎖定到事務(wù)結(jié)束(也就是RR級(jí)別,當(dāng)然MySQL使用MVCC在RC級(jí)別就解決了這個(gè)問題)
幻讀:當(dāng)同一個(gè)查詢?cè)诓煌瑫r(shí)間生成不同的行集合時(shí)就是出現(xiàn)了幻讀,針對(duì)的是其他事務(wù)的insert操作,為了解決這個(gè)問題,鎖定整個(gè)表到事務(wù)結(jié)束(也就是S級(jí)別,當(dāng)然MySQL使用間隙鎖在RR級(jí)別就解決了這個(gè)問題)
網(wǎng)上很多文章提到幻讀和提交讀的時(shí)候,有的說幻讀包括了delete的情況,有的說delete應(yīng)該屬于提交讀的問題,那到底真相如何呢?我們實(shí)際來看下MySQL的官方文檔(如下)
The so-called phantom problem occurs within a transaction when the same query produces different sets of rows at different times. For example, if a SELECT) is executed twice, but returns a row the second time that was not returned the first time, the row is a “phantom” row.
https://dev.mysql.com/doc/refman/5.7/en/innodb-next-key-locking.html
可以看到,幻讀針對(duì)的是結(jié)果集前后發(fā)生變化,所以看起來delete的情況應(yīng)該歸為幻讀,但是我們實(shí)際分析下上面列出的標(biāo)準(zhǔn)SQL在RR級(jí)別的實(shí)現(xiàn)原理就知道,標(biāo)準(zhǔn)SQL的RR級(jí)別是會(huì)對(duì)查到的數(shù)據(jù)行加行共享鎖,所以這時(shí)候其他事務(wù)想刪除這些數(shù)據(jù)行其實(shí)是做不到的,所以在RR下,不會(huì)出現(xiàn)因delete而出現(xiàn)幻讀現(xiàn)象,也就是幻讀不包含delete的情況。
網(wǎng)上很多文章會(huì)說MVCC或者M(jìn)VCC+間隙鎖解決了幻讀問題,實(shí)際上MVCC并不能解決幻讀問題。如以下的例子:
begin; #假設(shè)users表為空,下面查出來的數(shù)據(jù)為空 select * from users; #沒有加鎖 #此時(shí)另一個(gè)事務(wù)提交了,且插入了一條id=1的數(shù)據(jù) select * from users; #讀快照,查出來的數(shù)據(jù)為空 update users set name='mysql' where id=1;#update是當(dāng)前讀,所以更新成功,并生成一個(gè)更新的快照 select * from users; #讀快照,查出來id為1的一條記錄,因?yàn)镸VCC可以查到當(dāng)前事務(wù)生成的快照 commit;
可以看到前后查出來的數(shù)據(jù)行不一致,發(fā)生了幻讀。所以說只有MVCC是不能解決幻讀問題的,解決幻讀問題靠的是間隙鎖。如下:
begin; #假設(shè)users表為空,下面查出來的數(shù)據(jù)為空 select * from users lock in share mode; #加上共享鎖 #此時(shí)另一個(gè)事務(wù)B想提交且插入了一條id=1的數(shù)據(jù),由于有間隙鎖,所以要等待 select * from users; #讀快照,查出來的數(shù)據(jù)為空 update users set name='mysql' where id=1;#update是當(dāng)前讀,由于不存在數(shù)據(jù),不進(jìn)行更新 select * from users; #讀快照,查出來的數(shù)據(jù)為空 commit; #事務(wù)B提交成功并插入數(shù)據(jù)
關(guān)于MySQL中事務(wù)隔離級(jí)別的實(shí)現(xiàn)原理是什么就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。