久久久久久久视色,久久电影免费精品,中文亚洲欧美乱码在线观看,在线免费播放AV片

<center id="vfaef"><input id="vfaef"><table id="vfaef"></table></input></center>

    <p id="vfaef"><kbd id="vfaef"></kbd></p>

    
    
    <pre id="vfaef"><u id="vfaef"></u></pre>

      <thead id="vfaef"><input id="vfaef"></input></thead>

    1. 站長(zhǎng)資訊網(wǎng)
      最全最豐富的資訊網(wǎng)站

      一文淺析MySQL怎么解決幻讀問(wèn)題

      MySQL是如何解決幻讀問(wèn)題的?下面本篇文章就來(lái)帶大家聊聊這個(gè)問(wèn)題,下面就來(lái)帶著問(wèn)題一起看看文章吧!

      一文淺析MySQL怎么解決幻讀問(wèn)題

      ??金不三,銀不四的高頻面試題中,MySQL的事務(wù)特性,隔離級(jí)別等問(wèn)題也是非常經(jīng)典八股文之一,面對(duì)此種問(wèn)題,估計(jì)絕大數(shù)小伙伴也是信手拈來(lái)的事情:

      事務(wù)特性(ACID)原子性Atomicity)、隔離性Isolation)、一致性Consistency)和持久性

      隔離級(jí)別讀取未提交READ UNCOMMITTED),讀取已提交READ COMMITTED),可重復(fù)讀REPEATABLE READ),可串行化SERIALIZABLE

      而每一種隔離級(jí)別導(dǎo)致的問(wèn)題有:

      • READ UNCOMMITTED隔離級(jí)別下,可能發(fā)生臟讀不可重復(fù)讀幻讀問(wèn)題
      • READ COMMITTED隔離級(jí)別下,可能發(fā)生不可重復(fù)讀幻讀問(wèn)題,但是不可以發(fā)生臟讀問(wèn)題
      • REPEATABLE READ隔離級(jí)別下,可能發(fā)生幻讀問(wèn)題,但是不可以發(fā)生臟讀不可重復(fù)讀的問(wèn)題
      • SERIALIZABLE隔離級(jí)別下,各種問(wèn)題都不可以發(fā)生

      對(duì)于MySQL InnoDB 存儲(chǔ)引擎的默認(rèn)支持的隔離級(jí)別是 REPEATABLE-READ(可重讀),從上面的SQL標(biāo)準(zhǔn)的四種隔離級(jí)別定義可知,REPEATABLE-READ(可重復(fù)讀)是不可以防止幻讀的,但是我們都知道,MySQL InnoDB存儲(chǔ)引擎是解決了幻讀問(wèn)題發(fā)生的,那他又是如何解決的呢?

      1. 行格式

      ??在進(jìn)入主題之前,我們先大致了解一下什么是行格式,這樣有助于我們理解下面的MVCC,行格式是表中的行記錄在磁盤(pán)的存放方式,Innodb存儲(chǔ)引擎總共有4種不同類(lèi)型的行格式:compact、redundant、dynamic、compress;雖然很很多行格式,但是在原理上,大體都相同,如下,為compact行格式:一文淺析MySQL怎么解決幻讀問(wèn)題??從圖中可以看出來(lái),一條完整的記錄其實(shí)可以被分為記錄的額外信息記錄的真實(shí)數(shù)據(jù)兩大部分,記錄的額外信息分別是變長(zhǎng)字段長(zhǎng)度列表、NULL值列表記錄頭信息,而記錄的真實(shí)數(shù)據(jù)除了我們自己定義的列之外,MySQL會(huì)為每個(gè)記錄添加一些默認(rèn)列,這些默認(rèn)列又稱(chēng)為隱藏列,具體列如下:

      列名 長(zhǎng)度 描述
      row_id 6個(gè)字節(jié) 行ID,唯一標(biāo)識(shí)一條記錄
      transaction_id 6個(gè)字節(jié) 事務(wù)ID
      roll_pointer 7個(gè)字節(jié) 回滾指針

      隱藏列的值不用我們操心,InnoDB存儲(chǔ)引擎會(huì)自己幫我們生成的,畫(huà)得再詳細(xì)一點(diǎn),compact行格式如下:

      一文淺析MySQL怎么解決幻讀問(wèn)題

      • transaction_id :事物id,當(dāng)事物對(duì)行記錄進(jìn)行修改時(shí),都會(huì)將本事物的事物id賦值到該列
      • roll_pointer:每次在對(duì)行記錄進(jìn)行改動(dòng)的時(shí)候,都會(huì)把舊版本的數(shù)據(jù)寫(xiě)入undolog日志,然后將roll_pointer 指向該undolog,所以該列相當(dāng)于一個(gè)指針,通過(guò)該列,可以找到修改之前的信息

      2. MVCC詳解

      2.1 版本鏈

      假設(shè)有一條記錄如下:一文淺析MySQL怎么解決幻讀問(wèn)題插入該記錄的事務(wù)id80roll_pointer 指針為NULL(為了便于理解,讀者可理解為指向?yàn)镹ULL,實(shí)際上roll_pointer第一個(gè)比特位就標(biāo)記著它指向的undo日志的類(lèi)型,如果該比特位的值為1時(shí),就代表著它指向的undo日志類(lèi)型為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;
      登錄后復(fù)制

      ??每次對(duì)記錄進(jìn)行改動(dòng),都會(huì)記錄一條undo日志,每條undo日志也都有一個(gè)roll_pointer屬性(INSERT操作對(duì)應(yīng)的undo日志沒(méi)有該屬性,因?yàn)樵撚涗洸](méi)有更早的版本),可以將這些undo日志都連起來(lái),串成一個(gè)鏈表,所以現(xiàn)在的情況就像下圖一樣:

      一文淺析MySQL怎么解決幻讀問(wèn)題??對(duì)該記錄每次更新后,都會(huì)將舊值放到一條undo日志中,就算是該記錄的一個(gè)舊版本,隨著更新次數(shù)的增多,所有的版本都會(huì)被roll_pointer屬性連接成一個(gè)鏈表,我們把這個(gè)鏈表稱(chēng)之為版本鏈,版本鏈的頭節(jié)點(diǎn)就是當(dāng)前記錄最新的值。另外,每個(gè)版本中還包含生成該版本時(shí)對(duì)應(yīng)的事務(wù)id

      2.2 ReadView

      ??對(duì)于數(shù)據(jù)庫(kù)的四種隔離級(jí)別:1)read uncommitted;2) read committed;3) REPEATABLE READ; 4)SERIALIZABLE;來(lái)說(shuō),READ UNCOMMITTED,每次讀取版本鏈的最新數(shù)據(jù)即可;SERIALIZABLE,主要是通過(guò)加鎖控制;而read committedREPEATABLE READ都是讀取已經(jīng)提交了的事物,所以對(duì)于這兩個(gè)隔離級(jí)別,核心問(wèn)題是版本鏈中,哪些事物是對(duì)當(dāng)前事物可見(jiàn);為了解決這個(gè)問(wèn)題,MySQL提出了read view 概念,其包含四個(gè)核心概念:

      • m_ids:生成read view 時(shí)候,活躍的事物id集合
      • min_trx_idm_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,這樣在訪問(wèn)某條記錄時(shí),只需要按照下邊的步驟判斷記錄的某個(gè)版本是否可見(jiàn):一文淺析MySQL怎么解決幻讀問(wèn)題

      • 當(dāng)記錄的事物id等于creator_trx_id的時(shí)候,說(shuō)明當(dāng)前事物正在訪問(wèn)自己修改的記錄,所以該版本可見(jiàn)
      • 如果被訪問(wèn)的版本事物id小于min_trx_id的時(shí)候,則說(shuō)明,在創(chuàng)建read view的時(shí)候,該事物已經(jīng)提交,該版本,對(duì)當(dāng)前事物可讀
      • 如果被訪問(wèn)的版本事物id大于或等于max_trx_id,則說(shuō)明創(chuàng)建該read view的時(shí)候,該說(shuō)明生成該版本記錄的事物id在生成Read view之后才開(kāi)啟,所以該版本不能被當(dāng)前事物可讀
      • 如果被訪問(wèn)的版本事物transaction_idm_ids集合中,說(shuō)明生成Read view的時(shí)候,該事物還是活躍的,還沒(méi)有被提交,則該版本不可以被訪問(wèn);如果不在,則說(shuō)明創(chuàng)建ReadView時(shí)生成該版本的事務(wù)已經(jīng)被提交,可以被訪問(wèn)

      注:讀事物的事物id為0

      MySQL中,READ COMMITTEDREPEATABLE READ隔離級(jí)別的的一個(gè)非常大的區(qū)別就是它們生成ReadView的時(shí)機(jī)不同:

      • READ COMMITTED —— 每次讀取數(shù)據(jù)前都生成一個(gè)ReadView
      • REPEATABLE READ —— 在第一次讀取數(shù)據(jù)時(shí)生成一個(gè)ReadView

      下面我們通過(guò)詳細(xì)例子來(lái)說(shuō)明,兩者有何不同:

      時(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行記錄的版本鏈如下:

      一文淺析MySQL怎么解決幻讀問(wèn)題在時(shí)間⑥中,因事務(wù)trx 200 執(zhí)行了事務(wù)的提交,id=1行記錄的版本鏈如下:

      一文淺析MySQL怎么解決幻讀問(wèn)題

      在時(shí)間⑤,事務(wù)trx 100執(zhí)行select語(yǔ)句時(shí)會(huì)先生成一個(gè)ReadView,ReadViewm_ids列表的內(nèi)容就是[100, 200],min_trx_id100max_trx_id201,creator_trx_id0,此時(shí),從版本鏈中選可見(jiàn)的記錄,版本鏈從上到下遍歷:因?yàn)間rade=40,trx_id值為100,在m_ids里,所以該記錄不可見(jiàn),同理,grade=20的也不見(jiàn)。繼續(xù)往下遍歷,grade=20,trx_id值為80,小于小于ReadView中的min_trx_id100,所以這個(gè)版本符合要求,返回給用戶(hù)的是等級(jí)為10的記錄。

      在時(shí)間⑧中,如果事務(wù)的隔離級(jí)別是READ COMMITTED,會(huì)單獨(dú)又生成一個(gè)ReadView,該ReadViewm_ids列表的內(nèi)容就是[200],min_trx_id200,max_trx_id201creator_trx_id0,此時(shí),從版本鏈中選可見(jiàn)的記錄,版本鏈從上到下遍歷:因?yàn)間rade=70,trx_id值為200,在m_ids里,所以該記錄不可見(jiàn),繼續(xù)往下遍歷,grade=40,trx_id值為100,小于ReadView中的min_trx_id200,所以這個(gè)版本是符合要求的,返回給用戶(hù)的是是等級(jí)為40的記錄。

      在時(shí)間⑧中,如果事務(wù)的隔離級(jí)別是 REPEATABLE READ,在時(shí)間⑧中,不會(huì)單獨(dú)生成一個(gè)ReadView,而是沿用時(shí)間5的ReadView,所以返回給用戶(hù)的等級(jí)是10。前后兩次select得到的是一樣的,這就是可重復(fù)讀的含義。

      3. 總結(jié)

      ??通過(guò)分析MVCC詳解部分,可以得出,基于MVCC,在RR隔離級(jí)別下,很好解決了幻讀問(wèn)題,但是我們知道,select for update是產(chǎn)生當(dāng)前讀,不再是快照讀,那么此種情況,MySQL又是怎么解決幻讀問(wèn)題的呢?基于時(shí)間問(wèn)題(整理畫(huà)圖的確需要花比較多的時(shí)間),此處先給結(jié)論,后面再分析在當(dāng)前讀的情況下,MySQL是怎么解決幻讀問(wèn)題:

      • 當(dāng)前讀 : 使用 Next-Key Lock(間隙鎖) 進(jìn)行加鎖來(lái)保證不出現(xiàn)幻讀

      對(duì)于間隙鎖是如何在當(dāng)前讀的情況下解決幻讀問(wèn)題的,感興趣朋友可加個(gè)關(guān)注,點(diǎn)個(gè)贊

      贊(0)
      分享到: 更多 (0)
      網(wǎng)站地圖   滬ICP備18035694號(hào)-2    滬公網(wǎng)安備31011702889846號(hào)