SQLException : 处理 2 个不同的表时出现死锁错误

Posted

技术标签:

【中文标题】SQLException : 处理 2 个不同的表时出现死锁错误【英文标题】:SQLException : Deadlock error when processing 2 different tables 【发布时间】:2021-02-26 08:29:32 【问题描述】:

我遇到了死锁错误:

事务(进程 ID 91)在锁定资源上与另一个进程死锁,并已被选为死锁牺牲品。重新运行事务。

图像死锁 (https://i.stack.imgur.com/rDyxP.png)

Sp 91 是存储 Katzkin.dbo.ItemsAirBags_DX_IdItem:

delete ItemsAirBags
from ItemsAirBags  
where IdItem = @IdItem

Sp291 是存储 Katzkin.dbo.Items_TX_Description:

select *
from items  
where description = @description or left(description,5) = @description

我从 redgate 下载的完整 deadlock.xml

 <deadlock>
  <victim-list>
    <victimProcess id="process366a0a5848" />
  </victim-list>
  <process-list>
    <process id="process366a0a5848" taskpriority="0" logused="6424" waitresource="KEY: 6:72057609466478592 (56dfc8bc1bd3)" waittime="134" ownerId="2620704196" transactionname="user_transaction" lasttranstarted="2021-01-25T10:48:53.610" XDES="0x4462dd4490" lockMode="U" schedulerid="11" kpid="14292" status="suspended" spid="91" sbid="0" ecid="0" priority="0" trancount="2" lastbatchstarted="2021-01-25T10:48:53.690" lastbatchcompleted="2021-01-25T10:48:53.687" lastattention="2021-01-25T08:32:04.567" clientapp=".Net SqlClient Data Provider" hostname="SQLSERVER" hostpid="17172" loginname="KatCa" isolationlevel="read committed (2)" xactid="2620704196" currentdb="6" lockTimeout="4294967295" clientoption1="671088672" clientoption2="128056">
      <executionStack>
        <frame procname="Katzkin.dbo.ItemsAirBags_DX_IdItem" line="2" stmtstart="136" stmtend="258" sqlhandle="0x030006006b1f922228e7730050ab000001000000000000000000000000000000000000000000000000000000">
delete ItemsAirBags from ItemsAirBags  
where IdItem = @IdIte    </frame>
      </executionStack>
      <inputbuf>
Proc [Database Id = 6 Object Id = 580001643]   </inputbuf>
    </process>
    <process id="process3dc4191468" taskpriority="0" logused="34720" waitresource="KEY: 6:72057609931325440 (b015ce3d2676)" waittime="62" ownerId="2620704065" transactionname="user_transaction" lasttranstarted="2021-01-25T10:48:53.500" XDES="0x3d45478490" lockMode="S" schedulerid="6" kpid="1500" status="suspended" spid="291" sbid="0" ecid="0" priority="0" trancount="1" lastbatchstarted="2021-01-25T10:48:53.760" lastbatchcompleted="2021-01-25T10:48:53.760" lastattention="2021-01-25T08:30:50.683" clientapp=".Net SqlClient Data Provider" hostname="SQLSERVER" hostpid="30552" loginname="KatCa" isolationlevel="read committed (2)" xactid="2620704065" currentdb="6" lockTimeout="4294967295" clientoption1="671088672" clientoption2="128056">
      <executionStack>
        <frame procname="Katzkin.dbo.Items_TX_Description" line="4" stmtstart="172" stmtend="348" sqlhandle="0x030006008513045381f88c002da3000001000000000000000000000000000000000000000000000000000000">
Select * from items  
where description=@description or left(description,5)=@descriptio    </frame>
      </executionStack>
      <inputbuf>
Proc [Database Id = 6 Object Id = 1392776069]   </inputbuf>
    </process>
  </process-list>
  <resource-list>
    <keylock hobtid="72057609466478592" dbid="6" objectname="Katzkin.dbo.ItemsAirBags" indexname="PK_ItemsAirBags" id="lock3ca39db480" mode="X" associatedObjectId="72057609466478592">
      <owner-list>
        <owner id="process3dc4191468" mode="X" />
      </owner-list>
      <waiter-list>
        <waiter id="process366a0a5848" mode="U" requestType="wait" />
      </waiter-list>
    </keylock>
    <keylock hobtid="72057609931325440" dbid="6" objectname="Katzkin.dbo.Items" indexname="PK_Items" id="lock3f52da2100" mode="X" associatedObjectId="72057609931325440">
      <owner-list>
        <owner id="process366a0a5848" mode="X" />
      </owner-list>
      <waiter-list>
        <waiter id="process3dc4191468" mode="S" requestType="wait" />
      </waiter-list>
    </keylock>
  </resource-list>
</deadlock>

我们在两张不同的桌子上,但导致了死锁,我不明白原因以及如何解决它。 关于如何解决这些问题的任何想法?

【问题讨论】:

您在 ItemsAirBags 表上有任何触发器吗? @Arvo :不,我在两个表上都没有任何触发器 @Arvo:它们之间的关系只是通过外键。 造成死锁的原因之一是需要调整的 SELECT 查询,由于将 LEFT 函数应用于列,共享锁建议进行全扫描。如果您在description 上有索引,则将查询重构为SELECT * FROM items WHERE description LIKE LEFT(@description, 5); 可能会限制所触及的行数,OPTION *RECOMPILE) 查询提示也可能会有所帮助。另外,最佳做法是避免使用SELECT *,而是指定一个显式列列表。 【参考方案1】:

正如死锁图所示,您有 2 个存储过程,它们都试图更新/访问相同的行,但由于某种原因,其中一个是首先更新/访问 ItemsAirBags 表,然后是 Items 表和其他存储过程以相反的顺序。这就是造成僵局的原因。如果您使两个存储过程都以相同的顺序更新/访问两个表,您将消除死锁。

当您处理父子情况时,有时需要先对您计划更新的子行进行更新锁定,然后再更新父行,然后再更新子行。

您希望确保始终以相同的顺序更新/访问您的表,因为这样第一个存储过程将阻塞第二个存储过程(这是您想要的)而不是死锁。

注意:这些存储过程中发生的事情肯定比代码显示的要多,因为就目前而言,不会出现死锁。不知何故,进程process3dc4191468ItemsAirBags 上有一个排他锁,这不是所示 SQL 的一部分。

【讨论】:

您的意思是有其他商店首先更新/访问 ItemsAirBags 表,然后是 Items 表。正是此图中存储的相反顺序会导致死锁。 不是其他存储过程,死锁图中显示的2。死锁非常复杂,但也许这可能会有所帮助dotnettutorials.net/lesson/sql-server-deadlock-examples

以上是关于SQLException : 处理 2 个不同的表时出现死锁错误的主要内容,如果未能解决你的问题,请参考以下文章

原因:java.sql.SQLException:无法删除表“投票”上的外键约束“FK336ctjyksuuwnpmffcogcdyet”引用的表“链接”

在一个查询 MYSQL 上从 2 个不同的表中计算 2 个不同的东西

高悬赏!如何实现在c#中写SQL查询2个不同的表插入2个不同的表的记录!

Spring JPA:如何在同一个请求中更新 2 个不同的“DataSource”中的 2 个不同的表?

从 2 个不同的表中选择

使用 laravel 将数据保存在 2 个不同的表中