【mysql】mysql疑難問題:實際場景解釋什么是排它鎖 當前讀 快照讀

注: 理解本文 前置需要掌握的基礎知識:事務隔離、鎖的概念、并發知識;
事務隔離 尤其是事務延伸問題 是個重難點,絕非八股文那幾句話就能說完的,在實際場景中,分析起來有一定難度
@author: csdn博主 孟秋與你

本文將徐徐漸進,由簡入深分析一個案例

我們先看一個最簡單的例子:

Test test = new Test();long id = 1L;test .setId(id);test .setDes(System.currentTimeMillis()+"描述");try {testMapper.insert(test );} catch (DuplicateKeyException e) {Test t1 = tTestTMapper.selectById(id);return t1;}

我們插入一個id = 1 的數據,如果發現數據庫存在了 那就返回數據庫已有的數據;

然而 表面人畜無害的代碼,在高并發場景下 都有可能出現問題

當我們給代碼加上事務注解時:

    @Transactionalpublic void test() {Test test = new Test();long id = 1L;test .setId(id);test .setDes(System.currentTimeMillis()+"描述");try {testMapper.insert(test );} catch (DuplicateKeyException e) {Test oldRes= testMapper.selectById(id);return oldRes;}// 模擬其它業務try {Thread.sleep(5500L);} catch (InterruptedException e) {e.printStackTrace();}}

重點來了
spring中事務是和線程綁定的,假設有A線程和B線程 同時執行insert 方法,實際只會有一個插入成功,我們假設A執行成功了,B執行失敗,B線程會進入catch模塊 并查找數據庫已有的數據。
我們可以思考一下:

  1. B線程是如何知道自己失敗了的?
  2. B線程能成功查詢到結果嗎?

A1:id是唯一的(唯一索引同理),當A線程插入id=1 的數據后,mysql會為它加上一把排它鎖,當B線程企圖insert的時候 會一直等待鎖釋放,假設線程A執行時間是6s 在6s后才將事務提交, 那么線程B在6秒后才會進入到catch模塊 而不是立即拋出異常

A2:線程B在這個例子中能獲取到結果;但是會有些隱藏的坑。

catch模塊 事務B能獲取到事務A提交的結果嗎?
這取決于事務B是當前讀還是快照讀,在上面代碼中,由于之前沒有建立快照 在catch中才開始查詢 所以可以查到A事務提交的結果。

快照讀解釋:事務開始時會建立一個快照,這個快照代表了事務開始時刻數據庫的狀態。當事務執行SELECT查詢時,它看到的是該事務開始時數據庫數據的一個一致視圖,而不是當前時刻可能還在被其他事務修改的數據。這意味著,即使其他事務對數據進行了修改并提交,正在進行快照讀的事務看到的仍然是它自己快照中的數據版本,除非那些修改在該事務自己的更新操作中可見。

實際業務中 代碼會更加復雜:

    @Transactionalpublic void test() {Test test = new Test();long id = 1L;test.setId(id);test.setDes(System.currentTimeMillis()+"");// 模擬查詢已有數據 實際業務中可能是service來回調用查詢方法Test exist= testMapper.selectById(id);// 省略其它業務代碼  如校驗、關聯查詢等try {testMapper.insert(test);} catch (DuplicateKeyException e) {exist= testTMapper.selectById(id);System.out.println("數據已存在"+testT1);if (testT1 == null) {System.out.println("==============================================================================");}}try {Thread.sleep(5500L);} catch (InterruptedException e) {e.printStackTrace();}}

此時,并發場景下 線程B獲取到的就為null了;
因為在 這行代碼中, 已經生成了一個快照 , 所以在catch模塊中,也是查詢這個快照結果,它并不能感知到A線程已經插入了數據
Test exist= testMapper.selectById(id);

梳理一下執行情況 如下圖:
在這里插入圖片描述

如果我們希望catch模塊能拿到A線程已提交的數據 可以將快照讀改成當前讀:

	 // catch 模塊中 快照讀修改為當前讀// Test t1 = tTestTMapper.selectById(id);Test t1 = tTestTMapper.selectForUpdateById(id);

selectForUpdateById 示例:
select * from test where id = 1 for update
for update給當前數據加鎖,也正因為它會加鎖 所以是當前讀(否則數據就不正確了)

總結本文知識點:

  1. mysql唯一鍵插入數據時 會加上排它鎖,其它線程會等待它的鎖釋放
    (注意 不是表級鎖,例如我們的例子 只是id=1的數據加鎖 不要誤解)
  2. 事務默認是快照讀,當已建立快照時 是不能感知到其它線程并發修改的
  3. 可以通過for update 改成當前讀(需要注意for update會加鎖 )

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/diannao/79711.shtml
繁體地址,請注明出處:http://hk.pswp.cn/diannao/79711.shtml
英文地址,請注明出處:http://en.pswp.cn/diannao/79711.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Python:使用web框架Flask搭建網站

Date: 2025.04.19 20:30:43 author: lijianzhan Flask 是一個輕量級的 Python Web 開發框架,以簡潔靈活著稱,適合快速構建中小型 Web 應用或 API 服務。以下是 Flask 的核心概念、使用方法和實踐指南 Flask 的核心特點: 輕量級 核心代碼僅約…

層次式架構核心:中間層的功能、優勢與技術選型全解析

層次式架構中的中間層是整個架構的核心樞紐,承擔著多種重要職責,在功能實現、優勢體現以及技術選型等方面都有豐富的內容,以下為你詳細介紹: 一、功能 1.業務邏輯處理 復雜規則運算:在許多企業級應用中,…

網絡--應用層自定義協議與序列化

目錄 4-1 應用層 4-2 重新理解 read、write、recv、send 和 tcp 為什么支持全雙工 4-3 開始實現 4-1 應用層 我們程序員寫的一個個解決我們實際問題 , 滿足我們日常需求的網絡程序 , 都是在應用 層 . 再談 " 協議 " 協議是一種 " 約定 ". socke…

fastlio用mid360錄制的bag包離線建圖,提示消息類型錯誤

我用mid360錄制的bag包,激光雷達的數據類型是sensor_msgs::PointCloud2,但是運行fast_lio中的mid360 launch文件,會報錯(沒截圖),顯示無法從livox_ros_driver2::CustomMsg轉換到sensor_msgs::PointCloud2。…

C# WinForm窗口TextBox控件只能輸入數字(包括小數)并且恢復Ctrl+C復制和Ctrl+V粘貼功能

1. 前言 最近在寫定GPS定位時,經緯度是用的double類型,并且經緯度的要求是小數點后最少6位,多了能達到17位,又遇到了常用的TextBox控件只能輸入數字、小數的功能,因為有一年多沒有寫程序,現在再來寫這些感…

【MySQL數據庫】數據類型

目錄 1,數據類型分類 2,bit類型 3,小數類型 3-1,float/double類型 3-2,decimal類型 4,字符串類型 4-1,char 4-2,varchar 5,日期和時間類型 6,enum和…

Spark-SQL核心編程2

路徑問題 相對路徑與絕對路徑:建議使用絕對路徑,避免復制粘貼導致的錯誤,必要時將斜杠改為雙反斜杠。 數據處理與展示 SQL 風格語法:創建臨時視圖并使用 SQL 風格語法查詢數據。 DSL 風格語法:使用 DSL 風格語法查詢…

pandas庫詳解

CONTENT 基本數據結構SeriesDataFrame 數據讀取與寫入讀取 CSV 文件寫入 CSV 文件 數據清洗處理缺失值數據類型轉換 數據操作索引與切片數據合并數據分組與聚合 數據可視化 基本數據結構 Series Series 屬于一維標記數組,由一組數據和對應的索引構成。 import pa…

黑馬商城(五)微服務保護和分布式事務

一、雪崩問題 二、雪崩-解決方案&#xff08;服務保護方案&#xff09; 請求限流&#xff1a; 線程隔離&#xff1a; 服務熔斷&#xff1a; 服務保護組件&#xff1a; 三、Sentinel 引入依賴&#xff1a; <!--sentinel--> <dependency><groupId>com.aliba…

洛谷P1312 [NOIP 2011 提高組] Mayan 游戲

題目 #算法/進階搜索 思路: 根據題意,我們可以知道,這題只能枚舉,剪枝,因此,我們考慮如何枚舉,剪枝. 首先,我們要定義下降函數down(),使得小木塊右移時,能夠下降到最低處,其次,我們還需要寫出判斷函數,判斷矩陣內是否有小木塊沒被消除.另外,我們還需要消除函數,將矩陣內三個相連…

基于Redis的3種分布式ID生成策略

在分布式系統設計中&#xff0c;全局唯一ID是一個基礎而關鍵的組件。隨著業務規模擴大和系統架構向微服務演進&#xff0c;傳統的單機自增ID已無法滿足需求。高并發、高可用的分布式ID生成方案成為構建可靠分布式系統的必要條件。 Redis具備高性能、原子操作及簡單易用的特性&…

Spotlight on Mysql詳細介紹

1. 版本............................................................................................................................................1 2. 使用介紹...............................................................................................…

背包 DP 詳解

文章目錄 背包DP01 背包完全背包多重背包二進制優化單調隊列優化 小結 背包DP 背包 DP&#xff0c;說白了就是往一個背包里扔東西&#xff0c;求最后的最大價值是多少&#xff0c;一般分為了三種&#xff1a;01 背包、完全背包和多重背包。而 01 背包則是一切的基礎。 01 背包…

二級評論列表-Java實現

二級評論列表是很常見的功能&#xff0c;文章記錄了新手用Java實現的具體邏輯。 整體實現邏輯是先用2個sql&#xff0c;分別查出兩層數據。然后用java在service中實現數據組裝&#xff0c;返給前端。這種實現思路好處是SQL簡潔&#xff0c;邏輯分明&#xff0c;便于維護。 一…

快速入手-基于python和opencv的人臉檢測

1、安裝庫 pip install opencv-python 如果下載比較卡的話&#xff0c;指向國內下載地址&#xff1a; pip3 install opencv-python -i https://pypi.tuna.tsinghua.edu.cn/simple 2、下載源碼 https://opencv.org/ windows11對應的版本下載&#xff1a; https://pan.baidu…

GitLab本地安裝指南

當前GitLab的最新版是v17.10&#xff0c;安裝地址&#xff1a;https://about.gitlab.com/install/。當然國內也可以安裝極狐GitLab版本&#xff0c;極狐GitLab 是 GitLab 中國發行版&#xff08;JH&#xff09;。極狐GitLab支持龍蜥&#xff0c;歐拉等國內的操作系統平臺。安裝…

OpenCv高階(六)——圖像的透視變換

目錄 一、透視變換的定義與作用 二、透視變換的過程 三、OpenCV 中的透視變換函數 1. cv2.getPerspectiveTransform(src, dst) 2. cv2.warpPerspective(src, H, dsize, dstNone, flagscv2.INTER_LINEAR, borderModecv2.BORDER_CONSTANT, borderValue0) 四、文檔掃描校正&a…

資源-又在網上淘到金了

前言&#xff1a; 本期再分享網上沖浪發現的特效/動畫/視頻資源網站。 一、基本介紹&#xff1a; mantissa.xyz&#xff0c;about作者介紹為&#xff1a;Midge “Mantissa” Sinnaeve &#xff08;米奇辛納夫&#xff09;是一位屢獲殊榮的藝術家和導演&#xff0c;提供動畫、…

Linux疑難雜惑 | 云服務器重裝系統后vscode無法遠程連接的問題

報錯原因&#xff1a;本地的known_hosts文件記錄服務器信息與現服務器的信息沖突了&#xff0c;導致連接失敗。 解決方法&#xff1a;找到本地的known_hosts文件&#xff0c;把里面的所有東西刪除后保存就好了。 該文件的路徑可以在報錯中尋找&#xff1a;比如我的路徑就是&a…

FFMPEG-視頻解碼-支持rtsp|rtmp|音視頻文件(低延遲)

本人親測解碼顯示對比延遲達到7到20毫秒之間浮動兼容播放音視頻文件、拉流RTSP、RTMP等網絡流 基于 Qt 和 FFmpeg 的視頻解碼播放器類,繼承自 QThread,實現了視頻流的解碼、播放控制、幀同步和錯誤恢復等功能 工作流程初始化階段: 用戶設置URL和顯示尺寸 調用play()啟動線程解…