InnoDB事務隔離級別

来源:http://www.cnblogs.com/qinpengming/archive/2016/09/16/5876623.html
-Advertisement-
Play Games

轉載於:http://blog.csdn.net/wudongxu/article/details/8623610 SQL標准定義了4類隔離級別,包括了一些具體規則,用來限定事務內外的哪些改變是可見的,哪些是不可見的。低級別的隔離級一般支持更高的併發處理,並擁有更低的系統開銷。 Read Uncom ...


轉載於:http://blog.csdn.net/wudongxu/article/details/8623610

 

   SQL標准定義了4類隔離級別,包括了一些具體規則,用來限定事務內外的哪些改變是可見的,哪些是不可見的。低級別的隔離級一般支持更高的併發處理,並擁有更低的系統開銷。

 

Read Uncommitted(讀取未提交內容)

   在該隔離級別,所有事務都可以看到其他未提交事務的執行結果。本隔離級別很少用於實際應用,因為它的性能也不比其他級別好多少。讀取未提交的數據,也被稱之為臟讀(Dirty Read)。

 

Read Committed(讀取提交內容)

   這是大多數資料庫系統的預設隔離級別(但不是MySQL預設的)。它滿足了隔離的簡單定義:一個事務只能看見已經提交事務所做的改變。這種隔離級別也支持所謂的不可重覆讀(Nonrepeatable Read),因為同一事務的其他實例在該實例處理其間可能會有新的commit,所以同一select可能返回不同結果。

 

Repeatable Read(可重讀)

   這是MySQL的預設事務隔離級別,它確保同一事務的多個實例在併發讀取數據時,會看到同樣的數據行。不過理論上,這會導致另一個棘手的問題:幻讀(Phantom Read)。簡單的說,幻讀指當用戶讀取某一範圍的數據行時,另一個事務又在該範圍內插入了新行,當用戶再讀取該範圍的數據行時,會發現有新的“幻影” 行。InnoDB和Falcon存儲引擎通過多版本併發控制(MVCC,Multiversion Concurrency Control 間隙鎖)機制解決了該問題。註:其實多版本只是解決不可重覆讀問題,而加上間隙鎖(也就是它這裡所謂的併發控制)才解決了幻讀問題。

 

Serializable(可串列化)

   這是最高的隔離級別,它通過強制事務排序,使之不可能相互衝突,從而解決幻讀問題。簡言之,它是在每個讀的數據行上加上共用鎖。在這個級別,可能導致大量的超時現象和鎖競爭。

   臟讀(Drity Read):某個事務已更新一份數據,另一個事務在此時讀取了同一份數據,由於某些原因,前一個RollBack了操作,則後一個事務所讀取的數據就會是不正確的。

   不可重覆讀(Non-repeatableread):在一個事務的兩次查詢之中數據不一致,這可能是兩次查詢過程中間插入了一個事務更新了原有的數據。

   幻讀(Phantom Read):當對某行執行插入或刪除操作,而該行屬於某個事務正在讀取的行的範圍時,會發生幻像讀問題。事務第一次讀的行範圍顯示出其中一行已不復存在於第二次讀或後續讀中,因為該行已被其它事務刪除。同樣,由於其它事務的插入操作,事務的第二次或後續讀顯示有一行已不存在於原始讀中。幻讀其實也應該算是一種不可重覆讀現象,只是它只是相對於insert和delete操作,而上面的不可重覆讀現象但註重的是update操作。這裡這樣稱呼的原因是insert的新的row是沒有版本信息的,它要通過一個範圍來確定。

註:這裡的客戶端也是一個事務。

set sessiontransaction isolation level read uncommitted;

set autocommit=0;

臟讀的演示:

客戶端1

1362053703_7072.JPG

圖1

 

客戶端2

1362053728_2825.JPG

圖2

 

   通過上面兩張圖可以清楚的看到當tx_isolation=readuncommitted的時候,當一個事務(事務1)修改了數據但並沒有提交的時候,另外的事務(事務2)是可以訪問到該修改的數據。而此時如果前一個事務又取消了之前的修改(rollback)的時候,那麼事務2得到的數據就是一個“臟”數據。下麵我們看一下read committed是否有會這種情況。

 

客戶端1

1362053742_2513.JPG

圖3

 

1362053756_4593.JPG

圖4

 

   通過圖3可以看到即客戶端2 update了(但還沒commit),此時客戶端1讀到的還是之前的數據,當客戶端2真正的commit之後,客戶端1才能讀到更新後的數據。另外通過圖3我們同時可以看到另一個現象,不可重覆讀,即在一個事務內(客戶端1)兩次讀取的數據不一致。下麵我們看一下在Repeatable Read級別下是否有該現象。

 

Repeatable Read可重覆讀

1362053777_5236.JPG

圖5

 

1362053791_7368.JPG

圖6

 

   通過圖5可以看到即使客戶端2commit之後但客戶端1還沒commit的話,那麼在該事務內的任何時候,訪問同一條記錄的結果都是一樣的。所以repeatable read不存在不可重覆讀問題。

 

幻讀:其實跟不可重覆讀差不多,只是幻讀指的是行記錄數(一個範圍),而不可重覆讀相對的是一條記錄。如在事務1里第一次讀取這個範圍的記錄數為10條,但另一個事務刪除了這個範圍內的某一條記錄,這時就會導致事務2的再次讀取與第一次讀取的結果不一致。Innodb通過MCVV+間隙鎖解決了這個問題。所以對於innodb在repeatable read等級上也不會出現幻讀。當前串列化更不會出現上面的問題,因為

 

事務串列化

1362053889_8247.JPG

圖7

 

1362053907_1927.JPG

圖8

 

   註:上面的註釋序號表示操作或結果的出現順序。事務1先於事務2執行。通過圖7,8可以看到當事務1 select了表tx_test之後,事務2的update被阻塞了(因為存在一個共用鎖),只有當事務1 commit之後update才真正的被執行,同時此時tx_test又被事務2所占有(排它鎖),所以客戶1(事務3) select也被阻塞了,只有在

事務2 commit之後才得以執行。

   MVCC是為了減少加鎖而引入了,從而來提高併發性。MVCC只工作在repeatable read和read commit兩個隔離級別。而read uncommitted則是每次只管讀最新的版本的數據行,serializable則會對每個讀取操作加鎖,所以也不需要MVCC。

  上面就是innodb支持的4種隔離級別,以及它們存在的問題。同時我們通過實驗驗證了這些問題。

 

參考資料:

《高性能mysql》第二版

 


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

-Advertisement-
Play Games
更多相關文章
  • SQL 是用於訪問和處理資料庫的標準的電腦語言。 什麼是 SQL? SQL 指結構化查詢語言 SQL 使我們有能力訪問資料庫 SQL 是一種 ANSI 的標準電腦語言 編者註:ANSI,美國國家標準化組織 什麼是 SQL? SQL 指結構化查詢語言 SQL 使我們有能力訪問資料庫 SQL 是一種 ...
  • 之前一直沒用到這個函數,因為一般的情況下分類比較少的情況下,比如 : 男 : 0 女 : 1 我們一般會這麼設計資料庫,前臺顯示的時候一般會用select 或者 redio value = "0" 或=“1”顯示這樣確實是比較方便 其實用decode函數直接就可以使用了 select decode( ...
  • 一、登錄 打開終端,輸入/usr/local/mysql/bin/mysql -u root -p 初次進入mysql,密碼為空。當出現mysql>提示符時,表示你已經進入mysql中。鍵入exit退出mysql。 二、更改Mysqlroot用戶密碼 更改mysql root 用戶密碼,在終端輸入/ ...
  • oracle使用java source調用外部程式 需求 Oracle調用第三方外部程式。Oracle使用sqluldr2快速導出大批量數據,然後用winrar壓縮後發送郵件。 原碼 java source create or replace and compile java source name... ...
  • Spark Streaming Spark Streaming 是Spark為了用戶實現流式計算的模型。 數據源包括Kafka,Flume,HDFS等。 DStream 離散化流(discretized stream), Spark Streaming 使用DStream作為抽象表示。是隨時間推移而 ...
  • 地址:http://www.sqlite.org/download.html組織形式可以看到source code是Amalgamation。真正的源碼在這裡什麼是Amalgamation下載源碼以後,打開時候這樣子的src文件夾里就是所有的代碼,大概有一百多個。那麼為啥還要有一個Amalga... ...
  • 在MongoDB(版本 3.2.9)中,數據的分發是指將collection的數據拆分成塊(chunk),分佈到不同的分片(shard)上,數據分發主要有2種方式:基於數據塊(chunk)數量的均衡分發和基於片鍵範圍(range)的定向分發。MongoDB內置均衡器(balancer),用於拆分塊和 ...
  • 索引是一種特殊的文件(InnoDB數據表上的索引是表空間的一個組成部分),它們包含著對數據表裡所有記錄的引用指針。註:[1]索引不是萬能的!索引可以加快數據檢索操作,但會使數據修改操作變慢。每修改數據記錄,索引就必須刷新一次。為了在某種程式上彌補這一缺陷,許 多SQL命令都有一個DELAY_KEY_ ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...