這篇文章主要介紹“MySQL如何解決幻讀問題”,在日常操作中,相信很多人在MySQL如何解決幻讀問題問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對(duì)大家解答”MySQL如何解決幻讀問題”的疑惑有所幫助!接下來,請(qǐng)跟著小編一起來學(xué)習(xí)吧!
創(chuàng)新互聯(lián)公司堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:成都網(wǎng)站設(shè)計(jì)、網(wǎng)站制作、企業(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è)合作伙伴!
事務(wù)特性(ACID):原子性
(Atomicity
)、隔離性
(Isolation
)、一致性
(Consistency
)和持久性
隔離級(jí)別:讀取未提交
(READ UNCOMMITTED
),讀取已提交
(READ COMMITTED
),可重復(fù)讀
(REPEATABLE READ
),可串行化
(SERIALIZABLE
)
而每一種隔離級(jí)別導(dǎo)致的問題有:
READ UNCOMMITTED
隔離級(jí)別下,可能發(fā)生臟讀
、不可重復(fù)讀
和幻讀
問題
READ COMMITTED
隔離級(jí)別下,可能發(fā)生不可重復(fù)讀
和幻讀
問題,但是不可以發(fā)生臟讀
問題
REPEATABLE READ
隔離級(jí)別下,可能發(fā)生幻讀
問題,但是不可以發(fā)生臟讀
和不可重復(fù)讀
的問題
SERIALIZABLE
隔離級(jí)別下,各種問題都不可以發(fā)生
對(duì)于MySQL InnoDB 存儲(chǔ)引擎的默認(rèn)支持的隔離級(jí)別是 REPEATABLE-READ(可重讀),從上面的SQL標(biāo)準(zhǔn)的四種隔離級(jí)別定義可知,REPEATABLE-READ(可重復(fù)讀)
是不可以防止幻讀的,但是我們都知道,MySQL InnoDB存儲(chǔ)引擎是解決了幻讀問題發(fā)生的,那他又是如何解決的呢?
??在進(jìn)入主題之前,我們先大致了解一下什么是行格式,這樣有助于我們理解下面的MVCC,行格式是表中的行記錄在磁盤的存放方式,Innodb
存儲(chǔ)引擎總共有4種不同類型的行格式:compact
、redundant
、dynamic
、compress
;雖然很很多行格式,但是在原理上,大體都相同,如下,為compact
行格式:??從圖中可以看出來,一條完整的記錄其實(shí)可以被分為記錄的額外信息
和記錄的真實(shí)數(shù)據(jù)
兩大部分,記錄的額外信息
分別是變長字段長度列表
、NULL值列表
和記錄頭信息
,而記錄的真實(shí)數(shù)據(jù)
除了我們自己定義的列之外,MySQL會(huì)為每個(gè)記錄添加一些默認(rèn)列,這些默認(rèn)列又稱為隱藏列
,具體列如下:
列名 | 長度 | 描述 |
---|---|---|
row_id | 6個(gè)字節(jié) | 行ID,唯一標(biāo)識(shí)一條記錄 |
transaction_id | 6個(gè)字節(jié) | 事務(wù)ID |
roll_pointer | 7個(gè)字節(jié) | 回滾指針 |
隱藏列的值不用我們操心,InnoDB
存儲(chǔ)引擎會(huì)自己幫我們生成的,畫得再詳細(xì)一點(diǎn),compact
行格式如下:
transaction_id :事物id,當(dāng)事物對(duì)行記錄進(jìn)行修改時(shí),都會(huì)將本事物的事物id賦值到該列
roll_pointer:每次在對(duì)行記錄進(jìn)行改動(dòng)的時(shí)候,都會(huì)把舊版本的數(shù)據(jù)寫入undolog日志,然后將roll_pointer
指向該undolog
,所以該列相當(dāng)于一個(gè)指針,通過該列,可以找到修改之前的信息
假設(shè)有一條記錄如下:插入該記錄的事務(wù)id
為80
,roll_pointer
指針為NULL(為了便于理解,讀者可理解為指向?yàn)镹ULL,實(shí)際上roll_pointer第一個(gè)比特位就標(biāo)記著它指向的undo日志的類型,如果該比特位的值為1時(shí),就代表著它指向的undo日志類型為insert undo)
假設(shè)之后兩個(gè)事務(wù)id
分別為100
、200
的事務(wù)對(duì)這條記錄進(jìn)行UPDATE
操作:
-- 事務(wù)id=100
update person set grade =20 where id =1;
update person set grade =40 where id =1;
-- 事務(wù)id=200
update person set grade =70 where id =1;
??每次對(duì)記錄進(jìn)行改動(dòng),都會(huì)記錄一條undo日志
,每條undo日志
也都有一個(gè)roll_pointer
屬性(INSERT
操作對(duì)應(yīng)的undo日志
沒有該屬性,因?yàn)樵撚涗洸]有更早的版本),可以將這些undo日志
都連起來,串成一個(gè)鏈表,所以現(xiàn)在的情況就像下圖一樣:
??對(duì)該記錄每次更新后,都會(huì)將舊值放到一條undo日志
中,就算是該記錄的一個(gè)舊版本,隨著更新次數(shù)的增多,所有的版本都會(huì)被roll_pointer
屬性連接成一個(gè)鏈表,我們把這個(gè)鏈表稱之為版本鏈
,版本鏈的頭節(jié)點(diǎn)就是當(dāng)前記錄最新的值。另外,每個(gè)版本中還包含生成該版本時(shí)對(duì)應(yīng)的事務(wù)id
??對(duì)于數(shù)據(jù)庫的四種隔離級(jí)別:1)read uncommitted
;2) read committed
;3) REPEATABLE READ
; 4)SERIALIZABLE
;來說,READ UNCOMMITTED
,每次讀取版本鏈的最新數(shù)據(jù)即可;SERIALIZABLE
,主要是通過加鎖控制;而read committed
和REPEATABLE READ
都是讀取已經(jīng)提交了的事物,所以對(duì)于這兩個(gè)隔離級(jí)別,核心問題是版本鏈中,哪些事物是對(duì)當(dāng)前事物可見;為了解決這個(gè)問題,MySQL提出了read view 概念,其包含四個(gè)核心概念:
m_ids
:生成read view
時(shí)候,活躍的事物id集合
min_trx_id
:m_ids的最小值
,既生成read view的時(shí)候,活躍事物的最小值
max_trx_id
:表示生成read view
的時(shí)候,系統(tǒng)應(yīng)該分配下一個(gè)事物id值
creator_trx_id
:創(chuàng)建read view
的事物id,即當(dāng)前事物id。
有了這個(gè)ReadView
,這樣在訪問某條記錄時(shí),只需要按照下邊的步驟判斷記錄的某個(gè)版本是否可見:
當(dāng)記錄的事物id等于creator_trx_id
的時(shí)候,說明當(dāng)前事物正在訪問自己修改的記錄,所以該版本可見
如果被訪問的版本事物id小于min_trx_id
的時(shí)候,則說明,在創(chuàng)建read view
的時(shí)候,該事物已經(jīng)提交,該版本,對(duì)當(dāng)前事物可讀
如果被訪問的版本事物id大于或等于max_trx_id
,則說明創(chuàng)建該read view
的時(shí)候,該說明生成該版本記錄的事物id在生成Read view
之后才開啟,所以該版本不能被當(dāng)前事物可讀
如果被訪問的版本事物transaction_id
在m_ids
集合中,說明生成Read view
的時(shí)候,該事物還是活躍的,還沒有被提交,則該版本不可以被訪問;如果不在,則說明創(chuàng)建ReadView
時(shí)生成該版本的事務(wù)已經(jīng)被提交,可以被訪問
注:讀事物的事物id為0
在MySQL
中,READ COMMITTED
和REPEATABLE READ
隔離級(jí)別的的一個(gè)非常大的區(qū)別就是它們生成ReadView的時(shí)機(jī)不同:
READ COMMITTED
—— 每次讀取數(shù)據(jù)前都生成一個(gè)ReadView
REPEATABLE READ
—— 在第一次讀取數(shù)據(jù)時(shí)生成一個(gè)ReadView
下面我們通過詳細(xì)例子來說明,兩者有何不同:
時(shí)間編號(hào) | trx 100 | trx 200 | |
---|---|---|---|
① | BEGIN; | ||
② | BEGIN; | BEGIN; | |
③ | update person set grade =20 where id =1; | ||
④ | update person set grade =40 where id =1; | ||
⑤ | SELECT * FROM person WHERE id = 1; | ||
⑥ | COMMIT; | ||
⑦ | update person set grade =70 where id =1; | ||
⑧ | SELECT * FROM person WHERE id = 1; | ||
⑨ | COMMIT; | ||
? | COMMIT; |
在時(shí)間④中,因事務(wù)trx 100
執(zhí)行了事務(wù)的提交,id=1行記錄的版本鏈如下:
在時(shí)間⑥中,因事務(wù)trx 200
執(zhí)行了事務(wù)的提交,id=1行記錄的版本鏈如下:
在時(shí)間⑤,事務(wù)trx 100
執(zhí)行select
語句時(shí)會(huì)先生成一個(gè)ReadView
,ReadView
的m_ids
列表的內(nèi)容就是[100, 200]
,min_trx_id
為100
,max_trx_id
為201
,creator_trx_id
為0
,此時(shí),從版本鏈中選可見的記錄,版本鏈從上到下遍歷:因?yàn)間rade=40,trx_id
值為100
,在m_ids
里,所以該記錄不可見,同理,grade=20的也不見。繼續(xù)往下遍歷,grade=20,trx_id
值為80
,小于小于ReadView
中的min_trx_id
值100
,所以這個(gè)版本符合要求,返回給用戶的是等級(jí)為10的記錄。
在時(shí)間⑧中,如果事務(wù)的隔離級(jí)別是READ COMMITTED
,會(huì)單獨(dú)又生成一個(gè)ReadView
,該ReadView
的m_ids
列表的內(nèi)容就是[200]
,min_trx_id
為200
,max_trx_id
為201
,creator_trx_id
為0
,此時(shí),從版本鏈中選可見的記錄,版本鏈從上到下遍歷:因?yàn)間rade=70,trx_id
值為200
,在m_ids
里,所以該記錄不可見,繼續(xù)往下遍歷,grade=40,trx_id
值為100
,小于ReadView
中的min_trx_id
值200
,所以這個(gè)版本是符合要求的,返回給用戶的是是等級(jí)為40的記錄。
在時(shí)間⑧中,如果事務(wù)的隔離級(jí)別是 REPEATABLE READ
,在時(shí)間⑧中,不會(huì)單獨(dú)生成一個(gè)ReadView
,而是沿用時(shí)間5的ReadView
,所以返回給用戶的等級(jí)是10。前后兩次select得到的是一樣的,這就是可重復(fù)讀
的含義。
到此,關(guān)于“MySQL如何解決幻讀問題”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)砀鄬?shí)用的文章!