MySQL到底有沒有解決幻讀問題?這篇文章徹底給你解答

来源:https://www.cnblogs.com/yidengjiagou/archive/2022/09/13/16688967.html
-Advertisement-
Play Games

MySQL InnoDB引擎在Repeatable Read(可重覆讀)隔離級別下,到底有沒有解決幻讀的問題? 網上眾說紛紜,有的說解決了,有的說沒解決,甚至有些大v的意見都無法達成統一。 今天就深入剖析一下,徹底解決這個幻讀的問題。 解決幻讀問題之前,先普及幾個知識點。 ...


MySQL InnoDB引擎在Repeatable Read(可重覆讀)隔離級別下,到底有沒有解決幻讀的問題?

網上眾說紛紜,有的說解決了,有的說沒解決,甚至有些大v的意見都無法達成統一。

今天就深入剖析一下,徹底解決這個幻讀的問題。

解決幻讀問題之前,先普及幾個知識點。

1. 併發事務產生的問題

先創建一張用戶表,用作數據驗證:

CREATE TABLE `user` (
  `id` int NOT NULL AUTO_INCREMENT COMMENT '主鍵',
  `name` varchar(100) DEFAULT NULL COMMENT '姓名',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB COMMENT='用戶表';

併發事務會產生下麵三個問題:

臟讀

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

image

從上面的示例圖中,可以看出,在事務2修改完數據,沒有提交的情況。事務1已經讀到事務2最新修改的數據,這種情況就屬於臟讀。

不可重覆讀

定義: 一個事務讀取到其他事務修改過的數據。

image

從上面的示例圖中,可以看出,在事務2修改完數據,並提交事務後。事務1第二次查詢已經讀到事務2最新修改的數據,這種情況就屬於不可重覆讀。

幻讀

定義: 一個事務讀取到其他事務最新插入的數據。

image

從上面的示例圖中,可以看出,在事務2插入完數據,並提交事務後。事務1第二次查詢已經讀到事務2最新插入的數據,這種情況就屬於幻讀。

2. 快照讀和當前讀

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

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

例如:select

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

例如:insert、update、delete、select for update、select lock in share mode。

3. 再談幻讀問題

MySQL在Repeatable Read(可重覆讀)隔離級別下,到底有沒有解決幻讀的問題?

只能說是部分解決了幻讀問題。

首先,在快照讀的情況下,是通過MVCC(復用讀視圖)解決了幻讀問題。

想詳細瞭解MVCC和讀視圖,可以翻一下上篇文章。

先手動設置一下MySQL的隔離級別為可重覆讀

SET SESSION TRANSACTION ISOLATION LEVEL REPEATABLE READ;

image

執行測試用例,驗證一下:

image

從上面的示例圖中,可以看出,事務1的兩次查詢,得到的結果一致,並沒有查到事務2最新插入的數據。

原因是,在可重覆讀隔離級別下,第一次快照讀的時候,生成了一個讀視圖。第二次快照讀的時候,復用了第一次生成的讀視圖,所以兩次查詢得到的結果一致。

所以,在快照讀的情況下,可重覆讀隔離級別是解決了幻讀的問題。

再測試一下,在當前讀的情況下,可重覆讀隔離級別是否解決幻讀問題:

image

從上面的示例圖中,可以看出,事務1的兩次查詢,得到的結果不一致。在事務2插入數據,並提交事務後。事務1的第二次執行當前讀(加了for update)的時候,讀到了事務2最新插入的數據。

原因是,在可重覆讀隔離級別下,每次執行當前讀會生成一個新的讀視圖,所以能讀到其他事務最新插入的數據。

所以,在當前讀的情況下,可重覆讀隔離級別是沒有解決了幻讀的問題。

在執行上面的測試用例的時候,我忽然想到一個問題,既然select for update的當前讀,出現了幻讀問題,是不是其他的當前讀也會復現幻讀問題,比如insert。

再執行測試用例,驗證一下:

image

跟預想的一樣,在insert當前讀的情況下,也出現了幻讀的問題(主鍵衝突)。

那有沒有什麼辦法?在可重覆讀隔離級別下,執行當前讀的時候,也能解決幻讀的問題?

當然有的,唯一的辦法就是加鎖

image

事務1在執行第一次查詢的時候,就對數據進行加鎖(使用for update),防止其他事務修改數據,這樣也就徹底解決了幻讀問題。

你覺得有什麼好辦法嗎?

image


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

-Advertisement-
Play Games
更多相關文章
  • 1.vim三種模式 | 模式 |操作 | | : : | : : | | 可視模式 | 可查看內容 | | 編輯模式 | 可查看可修改內容 | | 命令行模式 | 給vim發送控制命令,可查看內容 | 註:打開文件,預設是可視模式 2.三種模式的切換 可視模式下 按i/a/o鍵 >進入編輯模式 編輯 ...
  • 如果您還為數學計算的繁瑣,函數作圖的費事,所畫圖形的不規範二煩惱的話,那麼您真的需要這款Mathematica 13 for Mac(科學計算軟體),是Mac平臺上致力於科學計算的軟體,很好地結合了數值和符號計算引擎、圖形系統、編程語言、文本系統、和與其他應用程式的高級連接。很多功能在相應領域內處於 ...
  • ####1. whoami--查看當前登錄的用戶名 book@100ask:~/linux$ whoami book ####2. echo--列印命令,配合'>'或者'>>'使用 echo 列印信息 //輸出信息到終端 echo 列印信息 > 文件名 //先清空文件裡面的內容,然後將輸出信息保存到 ...
  • 作者:小牛呼嚕嚕 | https://xiaoniuhululu.com 電腦內功、JAVA底層、面試相關資料等更多精彩文章在公眾號「小牛呼嚕嚕 」 現代電腦系統 現代電腦系統與馮·諾依曼電腦差別不大,最大的區別馮·諾依曼電腦 是 以運算器為中心的,而現代電腦 以儲存器為中心: 我們主要 ...
  • 本篇為Redis性能問題診斷系列的第二篇,本文主要從應用發起的典型命令使用上進行講解,由於Redis為單線程服務架構,對於一些命令如果使用不當會極大的影響Redis的性能表現,這裡也會對不合理的使用方式給出優化解決方案。 ...
  • ChengYing是一站式全自動化全生命周期大數據平臺運維管家,提供大數據產品的一站式部署、運維、監控服務,其可實現產品部署、產品升級、版本回滾、擴縮節點、日誌診斷、集群監控、實時告警等功能,致力於最大化節省運維成本,降低線上故障率與運維難度,為客戶提供安全穩定的產品部署與監控。 ChengYing ...
  • 哈嘍兄弟們,中秋閑著沒事,整理了一些資料庫的基本操作,分享給大家,希望對大家有所幫助~ 一、SQL語句 (mysql 資料庫中的語言) show databases;查看資料庫 use "database_ name" ;進入資料庫 show tables; 查看當前資料庫中有哪些表 select ...
  • 柯煜昌 青雲科技研發顧問級工程師 目前從事 RadonDB 容器化研發,華中科技大學研究生畢業,有多年的資料庫內核開發經驗。 文章字數 3800+,閱讀時間 15 分鐘 背景 MySQL 5.7 的字典信息保存在非事務表中,並且存放在不同的文件中(.FRM,.PAR,.OPT,.TRN,.TRG 等 ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...