다음을 통해 공유

分析和防止 Azure SQL 数据库中的死锁

适用于:Azure SQL 数据库

本文介绍如何识别死锁,使用死锁图和查询存储来识别死锁中的查询,以及规划和测试更改以防止重复发生死锁。

本文重点介绍如何识别和分析锁争用导致的死锁。 详细了解可能发生死锁的资源中的其他类型的死锁。

死锁的发生方式

默认情况下,Azure SQL 数据库中的每个新数据库都启用了 读取提交的快照隔离(RCSI) 数据库设置。 RCSI 使用行版本控制提高并发。在启用了 RCSI 的情况下,读取数据的会话与写入数据的会话之间的阻塞可最大程度地减轻。 但是,Azure SQL 数据库中的数据库仍可能发生阻塞和死锁,因为:

  • 修改数据的查询可能会相互阻止。

  • 查询可能在导致更高阻塞的隔离级别下运行。 可以通过 Transact-SQL 中的客户端库方法、 查询提示SET TRANSACTION ISOLATION LEVEL 指定隔离级别。

  • RCSI 可能会禁用,导致数据库使用共享 (S) 锁来保护SELECT在已提交读取隔离级别下运行的语句。 这可能会增加阻塞和死锁。

示例死锁

当两个或多个任务因为每个任务都锁住了另一个任务试图锁定的资源而永久互相阻塞时,就会发生死锁。 死锁也称为循环依赖关系:在两任务死锁的情况下,事务 A 依赖于事务 B,事务 B 又依赖于事务 A,从而达成闭环。

例如:

  1. 会话 A 开始一个显式事务并运行一条 update 语句,该语句获取对表 SalesLT.Product 中一行的更新 (U) 锁,此锁将转换为排他 (X) 锁

  2. 会话 B 运行一条 update 语句,该语句修改 SalesLT.ProductDescription 表。 该更新语句连接到SalesLT.Product表,以查找要更新的正确行。

    • 会话 B 获取对 SalesLT.ProductDescription 表中 72 行的更新 (U) 锁。

    • 会话 B 需要对表 中的行(包括被会话 A 锁定的行)使用共享锁。会话 B 在 SalesLT.Product 上被阻塞。

  3. 会话 A 会继续其事务,并且现在会对 SalesLT.ProductDescription 表运行更新。 会话 A 被会话 B 在 上阻塞。

显示两个会话处于死锁状态的图示。每个会话都拥有对方进程需要才能继续的资源。

死锁中的所有事务将无限期等待,除非其中一个参与的事务被回滚,例如,因为其会话已终止。

数据库引擎死锁监视器定期检查陷入死锁的任务。 如果死锁监视器检测到循环依赖项,它将选择其中一个任务作为牺牲品,并终止其事务,并出现错误 1205:Transaction (Process ID <N>) was deadlocked on lock resources with another process and is chosen as the deadlock victim. Rerun the transaction. 以这种方式中断死锁可让死锁中的其他任务完成其事务。

注意

请在本文的死锁进程列表部分中详细了解选择死锁牺牲品的标准。

两个会话之间的死锁的关系图。一个会话被选为死锁牺牲品。

被选为死锁牺牲品的事务所属的应用程序应当重试该事务,该事务通常会在死锁中涉及的其他事务完成后成功执行。

最好在重试前引入一个简短的随机延迟,以避免再次遇到相同的死锁。 请详细了解如何设计针对暂时性错误的重试逻辑

Azure SQL 数据库中的默认隔离级别

默认情况下,Azure SQL 数据库中的新数据库会启用读取已提交的快照 (RCSI)。 RCSI 将“读取已提交的内容”隔离级别的行为更改为使用行版本控制来提供语句级一致性,而无需对 SELECT 语句使用共享 (S) 锁。

在启用了 RCSI 的情况下:

  • 读取数据的语句不会阻止修改数据的语句。
  • 修改数据的语句不会阻止读取数据的语句。

对于 Azure SQL 数据库中的新数据库,默认还会启用快照隔离级别。 快照隔离是一个附加的基于行的隔离级别,它为数据提供事务级一致性,并使用行版本来选择要更新的行。 若要使用快照隔离,查询或连接必须将其事务隔离级别显式设置为 SNAPSHOT。 只能在为数据库启用了快照隔离时执行此操作。

可以使用 Transact-SQL 确定是否启用了 RCSI 和/或快照隔离。 连接到 Azure SQL 数据库中的数据库并运行以下查询:

SELECT name,
       is_read_committed_snapshot_on,
       snapshot_isolation_state_desc
FROM sys.databases
WHERE name = DB_NAME();
GO

如果启用了 RCSI,则 is_read_committed_snapshot_on 列将返回值 1。 如果启用了快照隔离,则 snapshot_isolation_state_desc 列将返回值 ON

如果 Azure SQL 数据库中的某个数据库禁用 RCSI,请在重新启用 RCSI 之前调查禁用 RCSI 的原因。 应用程序代码可能期望读取数据的查询将被写入数据的查询阻塞,从而导致启用了 RCSI 时在争用条件下会产生错误结果。

解释死锁事件

Azure SQL 数据库中的死锁管理器检测到死锁并选择某个事务作为牺牲品后,会发出死锁事件。 换句话说,如果为死锁设置了警报,则会在解决单个死锁后触发通知。 无需为此死锁执行任何用户操作。 应用程序应具备重试逻辑,以便在收到错误 1205 后自动继续:Transaction (Process ID <N>) was deadlocked on lock resources with another process and has been chosen as the deadlock victim. Rerun the transaction.

但是,设置警报很有用,因为死锁可能再次发生。 死锁警报使你能够调查数据库中是否发生了重复死锁模式,在这种情况下,可以选择采取措施来防止死锁重新出现。 请在本文的监视与警报部分了解更多关于死锁警报功能的信息。

防止死锁的最佳方法

防止死锁再次发生的最低风险方法通常是优化非聚集索引以优化死锁中涉及的查询。

  • 此方法的风险较低,因为优化非聚集索引不需要更改查询代码本身,因此在重新编写导致错误数据返回给用户的 Transact-SQL 时会降低用户错误的风险。

  • 有效的非聚集索引优化有助于查询更高效地查找要读取和修改的数据。 通过减少查询需要访问的数据量,可以减少阻塞的可能性,并且通常可以防止死锁。

在某些情况下,创建或优化聚集索引可以减少阻塞和死锁。 由于聚集索引包含在所有非聚集索引定义中,因此,创建或修改聚集索引在具有现有非聚集索引的大型表上可能是 IO 密集型的耗时操作。 请详细了解聚集索引设计指南

在索引优化无法成功防止死锁时,可以使用其他方法:

  • 如果死锁仅在为死锁中所涉的某个查询选择特定执行计划时才会发生,那么通过查询存储来强制执行查询计划可能会防止死锁再次出现。

  • 为死锁中涉及的一个或多个事务重新编写 Transact-SQL 也可有助于防止死锁。 将显式事务分解为较小的事务需要仔细编码和测试,以确保发生并发修改时的数据有效性。

请在本文的防止死锁再次发生部分详细了解这些方法中的每一个。

监视死锁并针对死锁发出警报

在本文中,我们使用 AdventureWorksLT 示例数据库为死锁设置警报,导致示例死锁,分析示例死锁图,并测试更改以防止死锁重新出现。

本文使用 SQL Server Management Studio (SSMS) 客户端,因为它包含在交互式视觉模式下显示死锁图形的功能。 可以使用其他客户端(例如 Azure Data Studiosqlcmd 或你喜欢的 Transact-SQL 查询工具)来跟踪这些示例,但可能只能将死锁图形视为 XML。

创建 AdventureWorksLT 数据库

若要按照这些示例操作,请在 Azure SQL 数据库中创建新数据库,然后选择“示例”数据作为“数据源”。

有关如何使用 Azure 门户、Azure CLI 或 PowerShell 创建的 AdventureWorksLT 详细说明,请在快速入门中选择所选方法 :在 Azure SQL 数据库中创建单一数据库

在 Azure 门户中设置死锁警报

若要为死锁事件设置警报,请按照使用 Azure 门户为 Azure SQL 数据库和 Azure Synapse Analytics 创建警报一文中的步骤操作。

选择“死锁”作为警报的信号名称。 配置“操作组”以使用你选择的方法(例如“电子邮件/短信”操作类型)通知你。

使用扩展事件在 Azure SQL 数据库中收集死锁图

死锁图是有关死锁中涉及的进程和锁的丰富信息来源。 若要在 Azure SQL 数据库中使用扩展事件 (XEvents) 收集死锁图,请捕获 sqlserver.database_xml_deadlock_report 事件。

可以使用环形缓冲区目标事件文件目标通过 XEvents 收集死锁图。 下表汇总了有关选择适当目标类型的注意事项:

方法 优点 注意事项 使用场景
环形缓冲区目标 - 仅使用 Transact-SQL 进行简单设置。 - 当出于任何原因(例如脱机数据库或数据库故障转移)停止 XEvents 会话时,将会清除事件数据。

- 数据库资源用于维护环缓冲区中的数据和查询会话数据。
- 收集用于测试和学习的示例跟踪数据。

- 在无法立即使用事件文件目标设置会话时针对短期需求而创建。

- 在设置了自动化过程以将跟踪数据保存到表中时,用作跟踪数据的“着陆架”
事件文件目标 - 将事件数据保存到 Azure 存储中的 Blob,因此即使在会话停止后数据也可用。

- 可以从 Azure 门户或 Azure 存储资源管理器 下载事件文件并在本地进行分析,这不需要使用数据库资源来查询会话数据。
- 设置较为复杂,需要配置 Azure 存储容器和数据库范围凭证。 - 通常在以下情况下使用:你希望事件数据即使在事件会话停止后也仍保存。

- 你想要运行一个跟踪,该跟踪生成的事件数据量比要在内存中保留的要多。

选择要使用的目标类型:

环形缓冲区目标十分方便且易于设置,但容量有限,这可能会导致较旧的事件丢失。 环形缓冲区不会将事件保存到存储中,当 XEvents 会话停止时清除环形缓冲区目标。 这意味着,当数据库引擎出于任何原因(例如故障转移)重新启动时,收集的任何 XEvent 都不可用。 如果无法立即将 XEvents 会话设置为事件文件目标,则环缓冲区目标最适合学习和短期需求。

此示例代码创建一个 XEvents 会话,该会话使用环形缓冲区目标捕获内存中的死锁图。 环形缓冲区目标允许的最大内存为 4 MB,当数据库联机(例如在故障转移后)时,该会话自动运行。

若要为写入环形缓冲区目标的 sqlserver.database_xml_deadlock_report 事件创建并启动 XEvents 会话,请连接到数据库并运行以下 Transact-SQL:

CREATE EVENT SESSION [deadlocks] ON DATABASE
ADD EVENT sqlserver.database_xml_deadlock_report
ADD TARGET package0.ring_buffer
WITH
(
    STARTUP_STATE = ON,
    MAX_MEMORY = 4 MB
);
GO

ALTER EVENT SESSION [deadlocks] ON DATABASE
STATE = START;
GO

导致死锁

由于 优化锁定 始终在 Azure SQL 数据库中启用,因此死锁的可能性较小。 有关详细信息,以及优化锁定时可能发生的死锁示例,请参阅 “优化锁定”和“死锁”。

查看 XEvents 会话中的死锁图表

如果设置了 XEvents 会话以收集死锁,并且在会话启动后发生了死锁,则可查看以交互式图形显示的死锁图以及该死锁图的 XML。

可以使用不同的方法来获取环形缓冲区目标和事件文件目标的死锁信息。 选择已用于 XEvents 会话的目标:

如果设置了要写入环形缓冲区的 XEvents 会话,可以使用以下 Transact-SQL 查询死锁信息。 在运行该查询之前,请将 @tracename 的值替换为你的 XEvents 会话名称。

DECLARE @tracename AS sysname = N'deadlocks';

WITH ring_buffer
AS (SELECT CAST (target_data AS XML) AS rb
    FROM sys.dm_xe_database_sessions AS s
         INNER JOIN sys.dm_xe_database_session_targets AS t
             ON CAST (t.event_session_address AS BINARY (8)) = CAST (s.address AS BINARY (8))
    WHERE s.name = @tracename
          AND t.target_name = N'ring_buffer'),
 dx
AS (SELECT dxdr.evtdata.query('.') AS deadlock_xml_deadlock_report
    FROM ring_buffer
CROSS APPLY rb.nodes('/RingBufferTarget/event[@name=''database_xml_deadlock_report'']') AS dxdr(evtdata))
SELECT d.query('/event/data[@name=''deadlock_cycle_id'']/value').value('(/value)[1]', 'int') AS [deadlock_cycle_id],
       d.value('(/event/@timestamp)[1]', 'DateTime2') AS [deadlock_timestamp],
       d.query('/event/data[@name=''database_name'']/value').value('(/value)[1]', 'nvarchar(256)') AS [database_name],
       d.query('/event/data[@name=''xml_report'']/value/deadlock') AS deadlock_xml,
       LTRIM(RTRIM(REPLACE(REPLACE(d.value('.', 'nvarchar(2000)'), CHAR(10), ' '), CHAR(13), ' '))) AS query_text
FROM dx
CROSS APPLY deadlock_xml_deadlock_report.nodes('(/event/data/value/deadlock/process-list/process/inputbuf)') AS ib(d)
ORDER BY [deadlock_timestamp] DESC;
GO

以 XML 格式查看和保存死锁图

以 XML 格式查看死锁图让你可以复制死锁中涉及的 Transact-SQL 语句的 inputbuffer。 你还可以用文本格式分析死锁。

如果已使用 Transact-SQL 查询返回死锁图信息,那么,若要查看死锁图 XML,请从任意行选择 deadlock_xml 列中的值,以在 SSMS 的新窗口中打开死锁图的 XML。

此示例死锁图的 XML 为:

<deadlock>
  <victim-list>
    <victimProcess id="process24756e75088" />
  </victim-list>
  <process-list>
    <process id="process24756e75088" taskpriority="0" logused="6528" waitresource="KEY: 8:72057594045202432 (98ec012aa510)" waittime="192" ownerId="1011123" transactionname="user_transaction" lasttranstarted="2022-03-08T15:44:43.490" XDES="0x2475c980428" lockMode="U" schedulerid="3" kpid="30192" status="suspended" spid="89" sbid="0" ecid="0" priority="0" trancount="2" lastbatchstarted="2022-03-08T15:44:49.250" lastbatchcompleted="2022-03-08T15:44:49.210" lastattention="1900-01-01T00:00:00.210" clientapp="Microsoft SQL Server Management Studio - Query" hostname="LAPTOP-CHRISQ" hostpid="16716" loginname="chrisqpublic" isolationlevel="read committed (2)" xactid="1011123" currentdb="8" currentdbname="AdventureWorksLT" lockTimeout="4294967295" clientoption1="671096864" clientoption2="128056">
      <executionStack>
        <frame procname="unknown" queryhash="0xef52b103e8b9b8ca" queryplanhash="0x02b0f58d7730f798" line="1" stmtstart="2" stmtend="792" sqlhandle="0x02000000c58b8f1e24e8f104a930776e21254b1771f92a520000000000000000000000000000000000000000">
unknown    </frame>
      </executionStack>
      <inputbuf>
    UPDATE SalesLT.ProductDescription SET Description = Description
        FROM SalesLT.ProductDescription as pd
        JOIN SalesLT.ProductModelProductDescription as pmpd on
            pd.ProductDescriptionID = pmpd.ProductDescriptionID
        JOIN SalesLT.ProductModel as pm on
            pmpd.ProductModelID = pm.ProductModelID
        JOIN SalesLT.Product as p on
            pm.ProductModelID=p.ProductModelID
        WHERE p.Color = 'Red'   </inputbuf>
    </process>
    <process id="process2476d07d088" taskpriority="0" logused="11360" waitresource="KEY: 8:72057594045267968 (39e18040972e)" waittime="2641" ownerId="1013536" transactionname="UPDATE" lasttranstarted="2022-03-08T15:44:46.807" XDES="0x2475ca80428" lockMode="S" schedulerid="2" kpid="94040" status="suspended" spid="95" sbid="0" ecid="0" priority="0" trancount="2" lastbatchstarted="2022-03-08T15:44:46.807" lastbatchcompleted="2022-03-08T15:44:46.760" lastattention="1900-01-01T00:00:00.760" clientapp="Microsoft SQL Server Management Studio - Query" hostname="LAPTOP-CHRISQ" hostpid="16716" loginname="chrisqpublic" isolationlevel="read committed (2)" xactid="1013536" currentdb="8" currentdbname="AdventureWorksLT" lockTimeout="4294967295" clientoption1="671088672" clientoption2="128056">
      <executionStack>
        <frame procname="unknown" queryhash="0xef52b103e8b9b8ca" queryplanhash="0x02b0f58d7730f798" line="1" stmtstart="2" stmtend="798" sqlhandle="0x020000002c85bb06327c0852c0be840fc1e30efce2b7c8090000000000000000000000000000000000000000">
unknown    </frame>
      </executionStack>
      <inputbuf>
    UPDATE SalesLT.ProductDescription SET Description = Description
        FROM SalesLT.ProductDescription as pd
        JOIN SalesLT.ProductModelProductDescription as pmpd on
            pd.ProductDescriptionID = pmpd.ProductDescriptionID
        JOIN SalesLT.ProductModel as pm on
            pmpd.ProductModelID = pm.ProductModelID
        JOIN SalesLT.Product as p on
            pm.ProductModelID=p.ProductModelID
        WHERE p.Color = 'Silver';   </inputbuf>
    </process>
  </process-list>
  <resource-list>
    <keylock hobtid="72057594045202432" dbid="8" objectname="9e011567-2446-4213-9617-bad2624ccc30.SalesLT.ProductDescription" indexname="PK_ProductDescription_ProductDescriptionID" id="lock2474df12080" mode="U" associatedObjectId="72057594045202432">
      <owner-list>
        <owner id="process2476d07d088" mode="U" />
      </owner-list>
      <waiter-list>
        <waiter id="process24756e75088" mode="U" requestType="wait" />
      </waiter-list>
    </keylock>
    <keylock hobtid="72057594045267968" dbid="8" objectname="9e011567-2446-4213-9617-bad2624ccc30.SalesLT.Product" indexname="PK_Product_ProductID" id="lock2474b588580" mode="X" associatedObjectId="72057594045267968">
      <owner-list>
        <owner id="process24756e75088" mode="X" />
      </owner-list>
      <waiter-list>
        <waiter id="process2476d07d088" mode="S" requestType="wait" />
      </waiter-list>
    </keylock>
  </resource-list>
</deadlock>

若要将死锁图保存为 XML 文件,请执行以下操作:

  1. 依次选择“文件”和“另存为...”。
  2. 将“保存类型”值保留为默认的“XML 文件(*.xml)”
  3. 将“文件名”设置为你选择的名称。
  4. 选择“保存” 。

将死锁图保存为可在 SSMS 中以交互方式显示的 XDL 文件

查看死锁图的交互式表示形式有助于快速概览死锁中涉及的进程和资源以及快速识别死锁牺牲品。

若要将死锁图保存为可由 SSMS 以图形方式显示的文件,请执行以下操作:

  1. 从任意行选择 deadlock_xml 列中的值,以在 SSMS 的新窗口中打开死锁图的 XML。

  2. 依次选择“文件”和“另存为...”。

  3. 将“保存类型”设置为“所有文件”

  4. 文件名 设置为所选名称,扩展名设置为 .xdl

  5. 选择“保存” 。

    将死锁图形 XML 文件保存到具有 xsd 扩展名的文件的 SSMS 中的屏幕截图。

  6. 通过选择窗口顶部选项卡上的“X”或依次选择“文件”、“关闭”来关闭文件。

  7. 通过依次选择“文件”、“打开”、“文件”,在 SSMS 中重新打开该文件。 选择使用 .xdl 扩展名保存的文件。

    死锁图现在显示在 SSMS 中,其中包含死锁中涉及的进程和资源的可视表示形式。

    SSMS中打开的XDL文件的屏幕截图。死锁图形通过图示方式展示,其中进程用椭圆表示,锁定资源用矩形表示。

分析 Azure SQL 数据库的死锁

死锁图通常具有三个节点:

  • 受害者名单。 死锁牺牲品进程标识符。

  • Process-list。 死锁中涉及的全部进程的信息。 死锁图使用术语“进程”来表示运行事务的会话。

  • 资源列表。 死锁中涉及的资源的信息。

在分析死锁时,逐步查看这些节点很有用。

死锁牺牲品列表

死锁牺牲品列表显示已选为死锁牺牲品的进程。 在死锁图的可视表示形式中,进程由椭圆表示。 表示死锁牺牲品进程的椭圆上绘制了一个“X”。

直观显示死锁的屏幕截图。椭圆代表被选为牺牲品的进程,上面绘制了一个 X。

死锁图的 XML 视图中,victim-list 节点为已成为死锁牺牲品的进程提供了一个 ID。

在我们的示例死锁中,受害者进程 ID 为 process24756e75088。 在检查 process-list 和 resource-list 节点以了解有关牺牲品进程及其锁定或请求锁定的资源的详细信息时,我们可以使用此 ID。

死锁进程列表

死锁进程列表是有关死锁所涉及的事务的丰富信息来源。

死锁图的图形表示形式仅显示死锁图 XML 中包含的信息的子集。 死锁图中的椭圆表示进程,并显示包括以下内容的信息:

  • 会话 ID,也称为 SPID。

  • 会话的死锁优先级。 如果两个会话的死锁优先级不同,则会选择优先级较低的会话作为死锁牺牲品。 在此示例中,这两个会话具有相同的死锁优先级。

  • 会话使用的事务日志量(以字节为单位)。 如果两个会话的死锁优先级相同,则死锁监视器会将回滚的开销较低的会话选为死锁牺牲品。 该成本是通过比较各事务此时已写入的日志字节数来确定的。

    在我们的示例死锁中, session_id 89 使用了较少的事务日志,并被选为死锁受害者。

此外,将鼠标悬停在每个进程上,可以查看发生死锁前每个会话中运行的最后一个语句的输入缓冲区。 输入缓冲区显示在工具提示中。

在 SSMS 中直观显示死锁图的屏幕截图。两个椭圆表示进程。将显示一个进程的输入缓冲区。

还提供了有关死锁图的 XML 视图中的进程的其他信息,包括:

  • 会话的标识信息,例如客户端名称、主机名和登录名。

  • 发生死锁之前每个会话运行的最后一个语句的查询计划哈希。 查询计划哈希有助于从查询存储检索有关查询的详细信息。

在我们的示例死锁中:

  • 可以看到,这两个会话都是在 chrisqpublic 登录名下运行,并使用了 SSMS 客户端。

  • 在发生死锁之前,死锁牺牲品运行的最后一个语句的查询计划哈希是0x02b0f58d7730f798。 可以在输入缓冲区中看到此语句的文本。

  • 死锁中的另一个会话运行的最后一个语句的查询计划哈希也是0x02b0f58d7730f798。 可以在输入缓冲区中看到此语句的文本。 在这种情况下,这两个查询具有相同的查询计划哈希,因为除了用作相等谓词的文本值之外,这两个查询是相同的。

本文中稍后会使用这些值在查询存储中查找其他信息

死锁进程列表中的输入缓冲区的限制

对于死锁进程列表中的输入缓冲区信息,需要注意一些限制。

查询文本可能会在输入缓冲区中截断。 输入缓冲区限制为正在执行的语句的前 4000 个字符。

此外,死锁图中涉及的某些语句可能不包含在死锁图中。 在我们的示例中,会话 A 在单个事务中运行了两条 update 语句。 死锁图中仅包括第二条 update 语句(导致死锁的 update 语句)。 会话 A 运行的第一个更新语句通过阻止会话 B 在死锁中发挥了作用。死锁图中不包括query_hash 运行的第一个语句的输入缓冲区和相关信息。

若要识别死锁中涉及的多语句事务中完整运行的 Transact-SQL,可以在执行查询的存储过程或应用程序代码中找到相关信息,或当死锁发生时使用 扩展事件 运行追踪以捕获参与死锁事务的会话所运行的完整语句。 如果死锁中涉及的语句被截断,并且仅部分 Transact-SQL 显示在输入缓冲区中,你可以在查询存储中查找该语句的 Transact-SQL 及其执行计划

死锁资源列表

死锁资源列表显示死锁中的进程拥有和等待哪些锁资源。

资源由死锁的可视表示形式中的矩形表示:

在 SSMS 中直观显示的死锁图的屏幕截图。矩形显示死锁涉及的资源。

注意

数据库名称在 Azure SQL 数据库中的数据库死锁图中表示为 GUID(uniqueidentifier)。 这是 physical_database_namesys.dm_user_db_resource_governance 动态管理视图中列出的数据库的

在此示例死锁中:

  • 死锁牺牲品(我们称之为会话 A):

    • PK_Product_ProductID 表的 SalesLT.Product 索引上的键拥有排他 (X) 锁。

    • PK_ProductDescription_ProductDescriptionID 表的 SalesLT.ProductDescription 索引上的键请求更新 (X) 锁。

  • 另一个进程(我们称之为会话 B):

    • PK_ProductDescription_ProductDescriptionID 表的 SalesLT.ProductDescription 索引上的键拥有更新 (X) 锁。

    • PK_ProductDescription_ProductDescriptionID 表的 SalesLT.ProductDescription 索引上的键请求共享 (S) 锁。

可以在 resource-list 节点内的死锁图的 XML 中看到相同的信息。

在查询存储中查找查询执行计划

检查死锁涉及的语句的查询执行计划通常非常有用。 通常,可以使用死锁图进程列表 XML 视图中的查询计划哈希在查询存储中找到这些执行计划。

此 Transact-SQL 查询查找与已为示例死锁找到的查询计划哈希匹配的查询计划。 连接到 Azure SQL 数据库中的用户数据库并运行以下查询。

DECLARE @query_plan_hash AS BINARY (8) = 0x02b0f58d7730f798;

SELECT qrsi.end_time AS interval_end_time,
       qs.query_id,
       qp.plan_id,
       qt.query_sql_text,
       TRY_CAST (qp.query_plan AS XML) AS query_plan,
       qrs.count_executions
FROM sys.query_store_query AS qs
     INNER JOIN sys.query_store_query_text AS qt
         ON qs.query_text_id = qt.query_text_id
     INNER JOIN sys.query_store_plan AS qp
         ON qs.query_id = qp.query_id
     INNER JOIN sys.query_store_runtime_stats AS qrs
         ON qp.plan_id = qrs.plan_id
     INNER JOIN sys.query_store_runtime_stats_interval AS qrsi
         ON qrs.runtime_stats_interval_id = qrsi.runtime_stats_interval_id
WHERE query_plan_hash = @query_plan_hash
ORDER BY interval_end_time, query_id;
GO

你可能无法从查询存储中获取查询执行计划,具体取决于你的查询存储 CLEANUP_POLICY 或 QUERY_CAPTURE_MODE 设置。 在这种情况下,通常可以通过显示查询的估计执行计划来获取所需的信息。

查找增加阻塞的模式

检查死锁中涉及的查询执行计划时,请注意可能造成阻塞和死锁的模式。

  • 表或索引扫描。 在启用了 RCSI 的情况下运行修改数据的查询时,选择要更新的行是通过阻塞性扫描完成的。在此过程中,当读取数据值时,会在数据行上施加更新 (U) 锁。 如果数据行不符合更新条件,则会释放更新锁,并锁定并扫描下一行。

    通过优化索引来帮助修改查询更有效地查找行,可以减少发出的更新锁数。 这将减少阻塞和死锁的可能性。

  • 引用多个表的索引视图。 当你修改索引视图中引用的表时,数据库引擎还必须维护索引视图。 这需要释放更多的锁,并可能导致阻塞和死锁的增加。 索引视图也可能致使更新操作在“读取已提交的内容”隔离级别下内部执行。

  • 对外键约束中引用的列的修改。 修改约束中 FOREIGN KEY 引用的表中的列时,数据库引擎必须查找引用表中的相关行。 行版本不可用于这些读取。 如果启用了级联更新或删除,则隔离级别可能会在该语句的持续时间内升级为可序列化,以防止虚拟插入。

  • 锁提示。 查找其中指定的隔离级别需要更多锁的表提示。 这些提示包括 HOLDLOCK(相当于可序列化)、SERIALIZABLEREADCOMMITTEDLOCK(用于禁用 RCSI)和 REPEATABLEREAD。 此外,PAGLOCKTABLOCKUPDLOCKXLOCK 等提示可能会增加阻塞和死锁的风险。

    如果这些提示已到位,请研究实现这些提示的原因。 这些提示可能防止出现争用条件并确保数据有效性。 可以保留这些提示,并在必要时使用“ 防止死锁” 部分中的备用方法阻止将来的死锁。

    注意

    请在事务锁定和行版本控制指南中详细了解使用行版本控制修改数据时的行为。

检查某个事务的完整代码时(无论是在执行计划在还是在应用程序查询代码中),请查找其他有问题的模式:

  • 事务中的用户交互。 显式多语句事务中的用户交互会显著增加事务的持续时间。 这使得这些事务更有可能重叠,并且阻塞和死锁更有可能发生。

    同样,持有打开的事务以及在事务中查询不相关的数据库或系统会显著增加阻塞和死锁的可能性。

  • 按不同顺序访问对象的事务。 当并发显式多语句事务遵循相同的模式并按相同顺序访问对象时,死锁不太可能发生。

防止死锁再次发生

有多种方法可用于防止死锁再次发生,例如索引优化、使用查询存储强制执行计划以及修改 Transact-SQL 查询。

  • 查看表的聚集索引。 大多数表都可从聚集索引中受益,但表通常是意外被实现为的。

    检查聚集索引的一种方法是使用 sp_helpindex 系统存储过程。 例如,我们可以通过执行以下语句来查看 SalesLT.Product 表的索引摘要:

    EXECUTE sp_helpindex 'SalesLT.Product';
    GO
    

    查看index_description列。 一个表只能包含一个聚集索引。 如果为表实现了聚集索引,则 index_description 包含单词 clustered

    如果不存在聚集索引,则该表为堆。 在这种情况下,请检查该表是否是有意创建为堆,以解决特定的性能问题。 考虑根据聚集索引设计指南实现聚集索引。

    在某些情况下,创建或优化聚集索引可以减少或消除死锁中的阻塞。 在其他情况下,可以使用其他技术,例如此列表中的其他技术。

  • 创建或修改非聚集索引。 优化非聚集索引可以帮助你的修改查询更快地查找要更新的数据,从而减少所需的更新锁数。

    在我们的示例死锁中,在查询存储中找到的查询执行计划包含针对 PK_Product_ProductID 索引的聚集索引扫描。 死锁图指示此索引上的共享 (S) 锁等待是死锁中的组成部分。

    查询执行计划的屏幕截图。正在对 Product 表上的 PK_Product_ProductID 索引执行聚集索引扫描。

    正在执行此索引扫描,因为我们的更新查询需要修改名为 vProductAndDescription 的索引视图。 如本文 “查找增加阻塞的模式” 部分中所述,引用多个表的索引视图可能会增加阻塞和死锁的可能性。

    如果我们在 AdventureWorksLT 数据库中创建以下非聚集索引来“涵盖”索引视图引用的 SalesLT.Product 中的列,那么这会有助于查询更高效地查找行:

    CREATE INDEX IX_Product_ProductID_Name_ProductModelID
        ON SalesLT.Product(ProductID, Name, ProductModelID);
    GO
    

    创建此索引后,死锁不会再次发生。

    当死锁涉及对外键约束中引用的列的修改时,请确保FOREIGN KEY的引用表上的索引支持高效查找相关行。

    虽然索引在某些情况下可以显着提高查询性能,但索引也有开销和管理成本。 查看常规索引设计指南,以帮助在创建索引(尤其是宽索引和大型表上的索引)之前评估索引的好处。

  • 评估索引视图的值。 防止再次发生示例死锁的另一个选项是删除 SalesLT.vProductAndDescription 索引视图。 如果未使用该索引视图,这将减少一段时间内维护索引视图的开销。

  • 使用快照隔离。 在某些情况下,为死锁中涉及的一个或多个事务将事务隔离级别设置为快照可能会防止阻塞和死锁再次发生。

    SELECT时,对 语句使用此方法最有可能成功。 禁用了“读取已提交的快照”时,使用“读取已提交的内容”隔离级别的SELECT查询需要共享 (S) 锁。 在这些事务上使用快照隔离可消除对共享锁的需求,这可以防止阻塞和死锁。

    在启用了“读取提交的快照隔离”的数据库中,SELECT查询不需要共享 (S) 锁,因此在修改数据的事务之间更有可能发生死锁。 在多个修改数据的事务之间发生死锁的情况下,快照隔离可能导致更新冲突而不是死锁。 这同样需要其中一个事务重试其操作。

  • 使用查询存储强制执行某个计划。 你可能会发现死锁中的某个查询具有多个执行计划,且仅在使用特定计划时才会发生死锁。 可以通过在查询存储中强制执行某个计划来防止死锁再次发生。

  • 修改 Transact-SQL。 可能需要修改 Transact-SQL,才能防止死锁再次发生。 应谨慎执行修改 Transact-SQL,并对更改进行严格测试,以确保在并发运行这些修改时数据正确。 重新编写 Transact-SQL 时,请考虑:

    • 在事务中对语句进行排序,以便它们按相同的顺序访问对象。

    • 尽可能将事务分解为较小的事务。

    • 如有必要,请使用查询提示来优化性能。 可以在不更改应用程序代码的情况下使用查询存储来应用提示。

死锁指南中找到最大程度地减少死锁的其他方法。

注意

如果其中一个会话必须不重试就成功完成,或者当死锁涉及的某一查询无关紧要、且应始终选为牺牲品时,在某些情况下,你可以为一个或多个涉及死锁的会话调整死锁优先级。 虽然这不会阻止死锁重新出现,但它可能会减少未来死锁的影响。

删除 XEvents 会话

你可以让一个收集死锁信息的 XEvents 会话在关键数据库上长期运行。 如果你使用事件文件目标,这可能会导致产生大型文件(如果发生多个死锁)。 你可以从 Azure 存储中删除活动跟踪的 blob 文件(当前正在写入的文件除外)。

当你希望删除 XEvents 会话时,无论选择的目标类型如何,用于删除该会话的 Transact-SQL 都是相同的。

若要删除 XEvents 会话,请运行以下 Transact-SQL。 在运行代码之前,请把会话的名称替换为相应的值。

ALTER EVENT SESSION [deadlocks] ON DATABASE
STATE = STOP;
GO

DROP EVENT SESSION [deadlocks] ON DATABASE;
GO

使用 Azure 存储资源管理器

Azure 存储资源管理器是一个独立的应用程序,用于简化对 Azure 存储内的 blob 中存储的事件文件目标的处理。 可以使用存储资源管理器实现以下目的:

下载 Azure 存储资源管理器