SQL Server事務執行過程中中釋放鎖導致的死鎖問題 - 排查與分析

来源:https://www.cnblogs.com/maurrinho/archive/2023/05/05/17374160.html
-Advertisement-
Play Games

0. 前情提要 系統的某個用來上報數據的介面存在死鎖的問題。這個介面內部對多張表進行了Update操作,執行順序為A表、B表、C表、D表、A表。死鎖發生的SQL,一條是第一次更新A表的SQL,另一條是第二次更新A表的SQL。整個更新都處在一個事務內,理論上講,只要第一個Session開始執行事務,第 ...


0. 前情提要

系統的某個用來上報數據的介面存在死鎖的問題。這個介面內部對多張表進行了Update操作,執行順序為A表、B表、C表、D表、A表。死鎖發生的SQL,一條是第一次更新A表的SQL,另一條是第二次更新A表的SQL。整個更新都處在一個事務內,理論上講,只要第一個Session開始執行事務,第二個Session就會由於無法獲取到A表的鎖而被阻塞,直到第一個Session執行完畢釋放鎖,那為什麼對A表的更新還會產生死鎖呢?

1. 準備工作

用準備來做測試的表

SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE TABLE [dbo].[Test](
	[Id] [int] NOT NULL,
	[Progress] [int] NOT NULL,
	[Score] [int] NOT NULL,
	[Grade] [int] NOT NULL,
	[Usn] [int] NOT NULL,
	[ProjectId] [int] NOT NULL,
	[IsDeleted] [int] NOT NULL,
 CONSTRAINT [PK_Id] PRIMARY KEY CLUSTERED 
(
	[Id] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO
CREATE UNIQUE NONCLUSTERED INDEX [index_usn_and_project_id] ON [dbo].[Test]
(
	[Usn] ASC,
	[ProjectId] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, SORT_IN_TEMPDB = OFF, IGNORE_DUP_KEY = OFF, DROP_EXISTING = OFF, ONLINE = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
GO

插入一條測試用的數據

PRINT @@spid
insert into dbo.Test (Id, Progress, Score, Grade, Usn, ProjectId, IsDeleted)
VALUES (1, 1, 1, 1, 1, 1, 1);

2. 非聚簇索引的情況下,在表上執行一次update的加鎖過程

用來測試的update語句

BEGIN TRAN

PRINT @@spid
update dbo.Test
set Progress=2
WHERE Usn=1 and ProjectId=1

COMMIT TRAN

利用Sql Server Profiler追蹤加鎖與釋放鎖的過程

從圖中我們可以得知,在鎖增長階段,存在以下加鎖步驟:

  1. 根據輸入條件,找到非聚簇索引所在頁,並獲取非聚簇索引所在頁的IU鎖
  2. 獲取非聚簇索引的U鎖
  3. 根據非聚簇索引的值,找到聚簇索引(即主鍵索引)所在的頁,獲取頁的IU鎖
  4. 獲取聚簇索引的U鎖
  5. 將聚簇索引所在頁的鎖提升為IX鎖
  6. 將聚簇索引的鎖提升為X鎖
  7. 更新數據
  8. 釋放非聚簇索引所在頁的IU鎖
  9. 釋放非聚簇索引的U鎖
  10. 釋放聚簇索引的X鎖
  11. 釋放聚簇索引所在頁的IX鎖

所有的鎖都被爭取的申請與釋放

2. 非聚簇索引的情況下,在表上執行兩個update的加鎖

BEGIN TRAN

PRINT @@spid
update dbo.Test
set Progress=2
WHERE Usn=1 and ProjectId=1

update dbo.Test
set Score=2,Grade = 2
WHERE Usn=1 and ProjectId=1

COMMIT TRAN


從紅框中可以看到,非聚簇索引索引在第一個update時申請U鎖、釋放U鎖,併在第二個update時再次申請U鎖、釋放U鎖。

在這裡,有意思的事情就發生了:第一個update在聚簇索引上申請的X鎖,在第一個update執行完成後,並沒有像非聚簇索引一樣被釋放。

這裡猜測一個原因:由於非聚簇索引並不需要更新,它只是用來更新聚簇索引數據的一個跳板,因此在聚簇索引更新完後就被釋放了。而聚簇索引上的X鎖,則需要在事務commit之後才會被釋放。

為了驗證這一點,嘗試在第一次update時,同時更新ProjectId

BEGIN TRAN

PRINT @@spid
update dbo.Test
set Progress=2,ProjectId=2
WHERE Usn=1 and ProjectId=1

update dbo.Test
set Score=2,Grade = 2
WHERE Usn=1 and ProjectId=2

COMMIT TRAN


紅線上方的就是第一個update, 由於對非聚簇索引進行了修改,所以非聚簇索引頁申請了X鎖,且沒有在第一次Update語句結束後釋放鎖。由此驗證了上面的假設

3. 推測與模擬

由上面的內容,我們知道了一件事情:在Update時,會申請非聚簇索引上的U鎖,並且會在Update結束後(而不是事務提交後)釋放鎖;當在一個事務記憶體在兩個Update(且更新的是同一條數據)時,會在非聚簇索引上存在 獲取鎖 -> 釋放鎖 -> 獲取鎖 -> 釋放鎖 的過程。

同時,我們也知道,假如同時存在兩個Session要Update同一張表的同一條數據,那麼晚到的Session由於事務的原因,會等待第一個Session釋放非聚簇索引上的鎖(因為Update的入口就是獲取非聚簇索引的鎖)。

那麼,我們就能推理出一種情況:Session1執行完了第一個Update,並將自己持有的非聚簇索引的U鎖釋放,保留著聚簇索引的X鎖;Session2看到非聚簇索引的U鎖被釋放,於是獲取了U鎖;此時Session1試圖再次獲取非聚簇索引的U鎖,但非聚簇索引的U鎖已經被Session2占有。而Session2要進一步更新數據,就需要Session1手裡的聚簇索引的X鎖。於是,死鎖就發生了

接下來就是驗證:左側的sql先執行,在第一個update後,延遲1分鐘;左側sql開始之後,執行右側的sql。理論上講,此時左右會觸發死鎖:

最終的結果,成功觸發死鎖,右側的sql作為後來者被犧牲:

本次運行的加鎖、釋放鎖的過程,其中白色部分為左側sql的執行過程(spid=53),藍色的部分是右側的sql的執行過程(spid=57),下麵以53和57代替:

可以看到,53釋放了非聚簇索引後,57立刻獲取了非聚簇索引上的U鎖,並獲得了聚簇索引所在頁的IU鎖,但由於53任然持有聚簇索引的X鎖,導致57無法進一步申請聚簇索引的鎖:

而53這時試圖獲取非聚簇索引所在頁上的IU鎖(被57持有),由此引發了死鎖。經過一段時間後,資料庫發現死鎖的存在,並主動終結了57,讓57持有的資源被釋放,從而保證53可以順利執行

4. 結論

到這裡,本次死鎖發生的原因其實已經明確了,原因可以歸結為,在一個事務記憶體在對某一資源的重覆申請與釋放,在併發場景下,就可能會因為對該資源的爭奪而產生死鎖。在開發的過程中,應當儘量避免在併發情況下,對同一資源的重覆申請與釋放


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

-Advertisement-
Play Games
更多相關文章
  • 接上文 4、條件分支控制流 避免分支嵌套,異常放在代碼片段最前面 4.1、歸約函數 4.2、條件表達式的封裝避免過長而導致可讀性下降 4.3、德摩根定律 4.4、and、or優先順序 4.5、or短路效應 4.6、消失的分支 4.6.1、二分查找演算法 4.6.2、字典演算法 5、異常錯誤處理 無需多言 ...
  • ChatGPT很強大,可以幫我們處理很多問題,但這些問題的答案的正確性您是否有考證過呢? 昨晚,DD就收到了一個有趣的反饋: 提問:有什麼關於數據許可權設計的資料推薦嗎? ChatGPT居然介紹了一本根本不存在的書《數據許可權設計與實現》,作者居然還是我... 那麼你在使用ChatGPT的時候,有碰到過 ...
  • 本文介紹將Windows電腦中的Administrator、網路、回收站等系統自帶桌面圖標取消顯示或恢復顯示的方法。 在Windows10電腦中,一般在桌面上預設會顯示如下所示的一些系統自帶圖標。 然而,在上述這些圖標中,有一些我們可能相對而言使用的頻率比較低,比如網路圖標,以及上圖中最上面的Adm ...
  • 本文源碼部分基於內核 5.4 版本討論 在經過上篇文章 《從內核源碼看 slab 記憶體池的創建初始化流程》 的介紹之後,我們最終得到下麵這幅 slab cache 的完整架構圖: 本文筆者將帶大家繼續從內核源碼的角度繼續拆解 slab cache 的實現細節,接下來筆者會基於上面這幅 slab ca ...
  • 哈嘍大家好我是鹹魚,在《Linux 記憶體管理 pt.1》中我們學習了什麼是物理記憶體、虛擬記憶體,瞭解了記憶體映射、缺頁異常等內容 那麼今天我們來接著學習 Linux 記憶體管理中的多級頁表和大頁 多級頁表&大頁 在《Linux 記憶體管理 pt.1》中我們知道了內核為每個進程都維護了一張頁表,這張頁表用來記 ...
  • 在MySQL中,這幾個都是統計操作,很多人在使用的時候,都使用的是count(1),這有沒有問題?使用正確?達到了統計效果? 我們從效果和效率兩方面來分析下 執行效果 count(*) 包括了所有的列,在統計時不會忽略列值為null的數據count(1) 用1表示代碼行,在統計時不會忽略列值為nul ...
  • 4月20日,袋鼠雲成功舉行了以“數實融合,韌性生長”為主題的2023春季生長大會。會上重磅發佈了袋鼠雲生態伙伴計劃——“飛躍計劃2.0”,從商機、產品、聯合方案及數據業務服務層面,與合作伙伴強強聯手,共同打造數字化生態,同時在聯合營銷、渠道政策、賦能培訓、產品開放、技術服務、交付實施等方面全面升級夥 ...
  • 摘要:本文主要介紹GaussDB(DWS)網路流控能力,並對其管控效果進行驗證。 本文分享自華為雲社區《GaussDB(DWS)網路流控與管控效果》,作者:門前一棵葡萄樹。 上一篇博文GaussDB(DWS)網路調度與隔離管控能力,我們詳細介紹了GaussDB網路調度邏輯,並簡單介紹瞭如何應用網路隔 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...