一文詳解MySQL事務底層原理,全是乾貨,推薦收藏

来源:https://www.cnblogs.com/yidengjiagou/archive/2022/06/26/16413825.html
-Advertisement-
Play Games

迎面走來了你的面試官,身穿格子衫,挺著啤酒肚,髮際線嚴重後移的中年男子。 手拿泡著枸杞的保溫杯,胳膊夾著MacBook,MacBook上還貼著公司標語:“我愛加班”。 面試開始,直入正題。 面試官: 看你簡歷上面寫著精通MySQL,我先問你事務的特性是什麼? 老生常談,這個還有誰不會背的嗎? 我: ...


迎面走來了你的面試官,身穿格子衫,挺著啤酒肚,髮際線嚴重後移的中年男子。
手拿泡著枸杞的保溫杯,胳膊夾著MacBook,MacBook上還貼著公司標語:“我愛加班”。

面試開始,直入正題。

面試官: 看你簡歷上面寫著精通MySQL,我先問你事務的特性是什麼?

老生常談,這個還有誰不會背的嗎?

我: 這個我知道,事務有四大特性,分別是原子性(Atomicity)、一致性(Consistency)、隔離性(Isolation)、持久性(Durability),簡稱ACID。

原子性是指事務中所有操作要麼全部成功,要麼全部失敗。

一致性是指事務執行前後,數據始終處於一致性狀態,不會出現數據丟失。

隔離性是指事務提交前的中間狀態對其他事務不可見,即相互隔離。

持久性是指事務提交後,數據的修改永久保存在資料庫中。

面試官: 嗯,回答得不錯。那你知道MySQL底層是怎麼實現事務的四大特性?

這道題有點深了,需要背會redo log、undo log、mvcc。

千萬別說不知道這幾個東西是幹嘛用的。

不但要知道,還要跟事務扯上關係。

我: 原子性是undo log實現的,一致性是由代碼邏輯層面保證的,隔離性是由mvcc實現的,持久性是基於redo log實現的。

Redo Log(重做日誌):

Redo Log記錄的是物理日誌,也就是磁碟數據的修改。
用來保證服務崩潰後,仍能把事務中變更的數據持久化到磁碟上。

如果沒有Redo Log的話,會發生什麼?

修改數據的過程就變成這樣了:

  1. 從磁碟載入數據到記憶體
  2. 在記憶體中修改數據
  3. 把新數據持久化到磁碟

這樣做,會有嚴重的性能問題。

  1. InnoDB在磁碟中存儲的基本單元是頁,可能本次修改只變更一頁中幾個位元組,但是需要刷新整頁的數據,就很浪費資源。
  2. 一個事務可能修改了多頁中的數據,頁之間又是不連續的,就會產生隨機IO,性能更差。

所以為了提高寫入性能,於是就引入了Redo Log。

看一下引入Redo Log後修改流程:

  1. 從磁碟載入數據到記憶體
  2. 在記憶體中修改數據
  3. 把新數據寫到Redo Log Buffer
  4. Redo Log Buffer中數據持久化到Redo Log文件中
  5. Redo Log文件中數據持久化到資料庫磁碟中

Undo Log(回滾日誌):

Undo Log記錄的是邏輯日誌,用來回滾事務時,恢復到修改前的數據。

比如:當我們執行一條insert語句時,Undo Log就記錄一條相反的delete語句。

加入Undo Log之後的修改流程就是這樣的:

MVCC(多版本併發控制,Multi-Version Concurrency Control):

記錄的是某個時間點上的數據快照,用來實現不同事務之間數據的隔離性。

提到隔離性,一定要說一下事務的隔離級別。

說事務隔離級別之前,必須要先說一下併發事務產生的問題:

臟讀: 一個事務讀到其他事務未提交的數據。

不可重覆讀: 多次讀取相同的數據,得到的結果集不一致,即讀到其他事務提交後的數據。

幻讀: 相同的查詢條件,多次讀取的結果不一致,即讀到其他事務提交後的數據。

不可重覆讀與幻讀的區別是: 不可重覆讀是讀到了其他事務執行update、delete後的數據,而幻讀是讀到其他事務執行insert後的數據。

隔離級別

Read UnCommitted(讀未提交): 讀到其他事務未提交的數據,會出現臟讀、不可重覆讀、幻讀。

Read Committed(讀已提交): 讀到其他事務已提交的數據,解決了臟讀,會出現不可重覆讀、幻讀。

Repeatable Read(可重覆讀): 相同的數據,多次讀取到的結果集一致。解決了不可重覆讀,還是會出現幻讀。

Serializable(串列化): 所有事務串列執行,解決了幻讀。

再談MVCC:

MVCC解決了讀寫衝突,實現了讀寫並行,提升了事務的性能。

由於Read UnCommitted隔離級別下,每次都讀取最新的數據。而Serializable隔離級別下,對所有讀取數據都加鎖。這兩種隔離級不需要MVCC,所以MVCC只在Read Committed和Repeatable Read兩種隔離級別下起作用。

MVCC的實現方式通過兩個隱藏列trx_id(最近一次提交事務的ID)和roll_pointer(上個版本的地址),建立一個版本鏈。併在事務中讀取的時候生成一個ReadView(讀視圖),在Read Committed隔離級別下,每次讀取都會生成一個讀視圖,而在Repeatable Read隔離級別下,只會在第一次讀取時生成一個讀視圖。

InnoDB如何解決幻讀的?

先普及一下快照讀和當前讀。

當前讀: 讀取數據的最新版本,並對數據進行加鎖。

例如:insert、update、delete、select for update

快照讀: 讀取數據的歷史版本,不對數據加鎖。

例如:select

在當前讀的情況下,是通過加鎖來解決幻讀。

在快照讀的情況下,是通過MVCC來解決幻讀。

面試官: 還得是你,就你總結的全。今天的面試先到這吧,下一面要問你MySQL的鎖,你準備一下。

本文知識點總結:

文章持續更新,可以微信搜一搜「 一燈架構 」第一時間閱讀更多技術乾貨。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • SpringBoot使用Redis教程 應用環境: 存放Token、.... 第一步: 添加Redis依賴 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-re ...
  • 📄前言 這個小項目源於github項目:✨50 projects 50 days, 這個項目包含了50個小型前端項目,適合學習了Html+Css+JavaScript但是還沒有學習框架的前端新手作為練習。 這裡是原項目的代碼實現👉擴展卡片 Expanding Cards 📝分析 📍佈局 卡片 ...
  • 本章是系列文章的第八章,用著色演算法進行寄存器的分配過程。 本文中的所有內容來自學習DCC888的學習筆記或者自己理解的整理,如需轉載請註明出處。周榮華@燧原科技 寄存器分配 寄存器分配是為程式處理的值找到存儲位置的問題 這些值可以存放到寄存器,也可以存放在記憶體中 寄存器更快,但數量有限 記憶體很多,但 ...
  • 目錄 一.簡介 二.效果演示 三.源碼下載 四.猜你喜歡 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 基礎 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 轉場 零基礎 O ...
  • 背景: 一般我們可以用HashMap做本地緩存,但是HashMap功能比較弱,不支持Key過期,不支持數據範圍查找等。故在此實現了一個簡易的本地緩存,取名叫fastmap。 功能: 1.支持數據過期 2.支持等值查找 3.支持範圍查找 4.支持key排序 實現思路: 1.等值查找採用HashMap2 ...
  • 詳細講解python爬蟲代碼,爬微博搜索結果的博文數據。 爬取欄位: 頁碼、微博id、微博bid、微博作者、發佈時間、微博內容、轉發數、評論數、點贊數。 爬蟲技術: 1、requests 發送請求 2、datetime 時間格式轉換 3、jsonpath 快速解析json數據 4、re 正則表達式提... ...
  • 為什麼要多線程下載 俗話說要以終為始,那麼我們首先要明確多線程下載的目標是什麼,不外乎是為了更快的下載文件。那麼問題來了,多線程下載文件相比於單線程是不是更快? 對於這個問題可以看下圖。 橫坐標是線程數,縱坐標是使用對應線程數下載對應文件時花費的時間,藍橙綠代表下載文件的大小,每個線程下載對應文件2 ...
  • 基礎知識 python是一門腳本語言,它是解釋執行的。 python使用縮進做為語法,而且python2環境下同一個py文件中不能同時存在tab和空格縮進,否則會出錯,建議在IDE中顯示縮進符。 python在聲明變數時不寫數據類型,可以type(xx)來獲取欄位的類型,然後可以int(),list ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...