sql server 事务锁死

zYiGJN 2018-05-09 10:39:18

BEGIN TRAN
declare @PREFIX nvarchar(50),@TABLENAME nvarchar(50)
set @PREFIX = '180509'; set @TABLENAME = 'T_HIS_REGISTER'
SET NOCOUNT ON
if ( exists (SELECT 1 FROM T_PK_SAAS WITH(NOLOCK) WHERE PREFIX = @PREFIX AND TABLENAME = @TABLENAME))
BEGIN
update T_PK_SAAS set sn = sn + 1 where PREFIX = @PREFIX AND TABLENAME = @TABLENAME
END
else
BEGIN
IF (EXISTS(SELECT 1 FROM T_PK_SAAS WITH(XLOCK) WHERE PREFIX = @PREFIX AND TABLENAME = @TABLENAME ))
BEGIN
update T_PK_SAAS set sn = sn + 1 where PREFIX = @PREFIX AND TABLENAME = @TABLENAME
END
ELSE
BEGIN
BEGIN TRY
INSERT INTO T_PK_SAAS ([FID], [TABLENAME], [PREFIX], [SERVERID], [SN], [CURDATE], [OPERATOR], [OPERATEDATE], [OPERATEDEPT], [UPFLAG], [MYROWID], [FYID])
VALUES ( @PREFIX + @TABLENAME, @TABLENAME, @PREFIX, NULL, '1', @PREFIX, NULL, GETDATE(), NULL, N'1',
NULL, NULL);
END TRY
BEGIN CATCH
ROLLBACK TRAN
END CATCH
END
END
select sn from T_PK_SAAS where PREFIX = @PREFIX AND TABLENAME = @TABLENAME

COMMIT TRAN



就和标题说的并发冲突是生成主键的存储过程会发生事务锁死
...全文
905 12 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
12 条回复
切换为时间正序
请发表友善的回复…
发表回复
xiaoxiangqing 2018-05-10
  • 打赏
  • 举报
回复
是不是不是按相同的方式运行语句
zYiGJN 2018-05-09
  • 打赏
  • 举报
回复
引用 6 楼 roy_88 的回复:
查看是否缺少索引,建上索引再试 e.g.
CREATE INDEX IX_T_PK_SAAS  ON T_PK_SAAS (PREFIX ,TABLENAME)INCLUDE(sn);
但是我看在sql执行的时候查看了update与select加的锁都是key锁
zYiGJN 2018-05-09
  • 打赏
  • 举报
回复
引用 4 楼 shoppo0505 的回复:
INSERT INTO T_PK_SAAS 时候,[FID] = @PREFIX + @TABLENAME,这个FID是PK?
中国风 2018-05-09
  • 打赏
  • 举报
回复
查看是否缺少索引,建上索引再试 e.g.
CREATE INDEX IX_T_PK_SAAS  ON T_PK_SAAS (PREFIX ,TABLENAME)INCLUDE(sn);
zYiGJN 2018-05-09
  • 打赏
  • 举报
回复
引用 1 楼 roy_88 的回复:
改为以下优化方法 e.g.
BEGIN TRY
    BEGIN TRAN;
    DECLARE @PREFIX    NVARCHAR(50)
          , @TABLENAME NVARCHAR(50);
    SET @PREFIX='180509';
    SET @TABLENAME='T_HIS_REGISTER';
    SET NOCOUNT ON;

    UPDATE T_PK_SAAS
    SET sn=sn+1
    WHERE PREFIX=@PREFIX
          AND TABLENAME=@TABLENAME;
    IF @@ROWCOUNT=0
        INSERT INTO T_PK_SAAS ([FID]
                             , [TABLENAME]
                             , [PREFIX]
                             , [SERVERID]
                             , [SN]
                             , [CURDATE]
                             , [OPERATOR]
                             , [OPERATEDATE]
                             , [OPERATEDEPT]
                             , [UPFLAG]
                             , [MYROWID]
                             , [FYID])
        VALUES (@PREFIX+@TABLENAME, @TABLENAME, @PREFIX, NULL, '1', @PREFIX, NULL, GETDATE(), NULL, N'1', NULL, NULL);


    SELECT sn
    FROM T_PK_SAAS
    WHERE PREFIX=@PREFIX
          AND TABLENAME=@TABLENAME;

    COMMIT TRAN;
END TRY
BEGIN CATCH
    SELECT ERROR_MESSAGE();
    ROLLBACK TRAN;
END CATCH;
<deadlock-list>
 <deadlock victim="process239236508">
  <process-list>
   <process id="process239236508" taskpriority="0" logused="268" waitresource="KEY: 5:72057594637189120 (e400f8433bb2)" waittime="2248" ownerId="322950029" transactionname="implicit_transaction" lasttranstarted="2018-05-09T09:22:20.683" XDES="0x13ad99950" lockMode="U" schedulerid="4" kpid="468" status="suspended" spid="153" sbid="0" ecid="0" priority="0" trancount="3" lastbatchstarted="2018-05-09T09:22:20.683" lastbatchcompleted="2018-05-09T09:22:20.683" clientapp="Microsoft JDBC Driver for SQL Server" hostname="iZhc0v6l21noh2Z" hostpid="0" loginname="sa" isolationlevel="read committed (2)" xactid="322950029" currentdb="5" lockTimeout="4294967295" clientoption1="673185824" clientoption2="128058">
    <executionStack>
     <frame procname="CYYFHIS.dbo.CrePk" line="8" stmtstart="490" stmtend="662" sqlhandle="0x03000500f4c3cf6874363901d9a800000100000000000000">
update T_PK_SAAS set sn = sn + 1  where PREFIX = @PREFIX AND TABLENAME = @TABLENAME     </frame>
     <frame procname="adhoc" line="1" stmtstart="130" sqlhandle="0x01000500c9cd9223e07c1aa5000000000000000000000000">
EXEC dbo.CrePk @P0,@P1,@P2 OUT     </frame>
     <frame procname="unknown" line="1" sqlhandle="0x000000000000000000000000000000000000000000000000">
unknown     </frame>
    </executionStack>
    <inputbuf>
(@P0 nvarchar(4000),@P1 nvarchar(4000),@P2 nvarchar(4000) OUTPUT)EXEC dbo.CrePk @P0,@P1,@P2 OUT                        </inputbuf>
   </process>
   <process id="process2455e2bc8" taskpriority="0" logused="560" waitresource="KEY: 5:72057594637189120 (861b25f966bb)" waittime="2054" ownerId="322950027" transactionname="implicit_transaction" lasttranstarted="2018-05-09T09:22:20.653" XDES="0x155bbf950" lockMode="S" schedulerid="2" kpid="11296" status="suspended" spid="98" sbid="0" ecid="0" priority="0" trancount="2" lastbatchstarted="2018-05-09T09:22:20.877" lastbatchcompleted="2018-05-09T09:22:20.877" clientapp="Microsoft JDBC Driver for SQL Server" hostname="iZhc0v6l21noh2Z" hostpid="0" loginname="sa" isolationlevel="read committed (2)" xactid="322950027" currentdb="5" lockTimeout="4294967295" clientoption1="673185824" clientoption2="128058">
    <executionStack>
     <frame procname="CYYFHIS.dbo.CrePk" line="6" stmtstart="280" stmtend="466" sqlhandle="0x03000500f4c3cf6874363901d9a800000100000000000000">
if ( exists (SELECT 1 FROM  T_PK_SAAS WHERE PREFIX = @PREFIX AND TABLENAME = @TABLENAME))     </frame>
     <frame procname="adhoc" line="1" stmtstart="130" sqlhandle="0x01000500c9cd9223e07c1aa5000000000000000000000000">
EXEC dbo.CrePk @P0,@P1,@P2 OUT     </frame>
     <frame procname="unknown" line="1" sqlhandle="0x000000000000000000000000000000000000000000000000">
unknown     </frame>
    </executionStack>
    <inputbuf>
(@P0 nvarchar(4000),@P1 nvarchar(4000),@P2 nvarchar(4000) OUTPUT)EXEC dbo.CrePk @P0,@P1,@P2 OUT                        </inputbuf>
   </process>
  </process-list>
  <resource-list>
   <keylock hobtid="72057594637189120" dbid="5" objectname="CYYFHIS.dbo.T_PK_SAAS" indexname="PK_T_PK_SAAS" id="lock2197d0980" mode="X" associatedObjectId="72057594637189120">
    <owner-list>
     <owner id="process2455e2bc8" mode="X"/>
    </owner-list>
    <waiter-list>
     <waiter id="process239236508" mode="U" requestType="wait"/>
    </waiter-list>
   </keylock>
   <keylock hobtid="72057594637189120" dbid="5" objectname="CYYFHIS.dbo.T_PK_SAAS" indexname="PK_T_PK_SAAS" id="lock19f182b80" mode="X" associatedObjectId="72057594637189120">
    <owner-list>
     <owner id="process239236508" mode="X"/>
    </owner-list>
    <waiter-list>
     <waiter id="process2455e2bc8" mode="S" requestType="wait"/>
    </waiter-list>
   </keylock>
  </resource-list>
 </deadlock>
</deadlock-list>
这是我获取死锁的时候的详细信息
zYiGJN 2018-05-09
  • 打赏
  • 举报
回复
你好,那我能请问一下,为什么一个存储过程内单表的操作,会造成事务锁死,这个看不懂,头疼
shoppo0505 2018-05-09
  • 打赏
  • 举报
回复
INSERT INTO T_PK_SAAS 时候,[FID] = @PREFIX + @TABLENAME,这个FID是PK?
中国风 2018-05-09
  • 打赏
  • 举报
回复
改为以下优化方法 e.g.
BEGIN TRY
    BEGIN TRAN;
    DECLARE @PREFIX    NVARCHAR(50)
          , @TABLENAME NVARCHAR(50);
    SET @PREFIX='180509';
    SET @TABLENAME='T_HIS_REGISTER';
    SET NOCOUNT ON;

    UPDATE T_PK_SAAS
    SET sn=sn+1
    WHERE PREFIX=@PREFIX
          AND TABLENAME=@TABLENAME;
    IF @@ROWCOUNT=0
        INSERT INTO T_PK_SAAS ([FID]
                             , [TABLENAME]
                             , [PREFIX]
                             , [SERVERID]
                             , [SN]
                             , [CURDATE]
                             , [OPERATOR]
                             , [OPERATEDATE]
                             , [OPERATEDEPT]
                             , [UPFLAG]
                             , [MYROWID]
                             , [FYID])
        VALUES (@PREFIX+@TABLENAME, @TABLENAME, @PREFIX, NULL, '1', @PREFIX, NULL, GETDATE(), NULL, N'1', NULL, NULL);


    SELECT sn
    FROM T_PK_SAAS
    WHERE PREFIX=@PREFIX
          AND TABLENAME=@TABLENAME;

    COMMIT TRAN;
END TRY
BEGIN CATCH
    SELECT ERROR_MESSAGE();
    ROLLBACK TRAN;
END CATCH;
zYiGJN 2018-05-09
  • 打赏
  • 举报
回复
引用 8 楼 roy_88 的回复:
测过执行开销时长没?你按照#1/ #6 方法改,出锁死锁的机率应该很低 你要加锁只能在事务开始前加上 持续锁+更新锁 这会造成阻塞,影响性能
我找到原因了就是因为并发先查询存在与否会获取s锁,然后都在转化x锁但是s锁都不释放导致锁死,但是优化的方法就是你说的先update在判断rowcount,谢谢版主大大,结帖了
中国风 2018-05-09
  • 打赏
  • 举报
回复
测过执行开销时长没?你按照#1/ #6 方法改,出锁死锁的机率应该很低 你要加锁只能在事务开始前加上 持续锁+更新锁 这会造成阻塞,影响性能
SQL Server 2005微软官方权威参考书.   公球公认SQL Server 2005 经典著作..   数据库“铁人”、微软MVP胡百敬先生鼎力推荐   微软SQL Server 总部Principal Group 项目经理朱凌志鼎力推荐   本书详细介绍了数据引擎的基础运作,包含了数据库的设定与数据实际在硬盘的摆放、索引结构、事务与锁定等。除了解释设计理念与运作原理外,还辅之以测试验证的方式。数据库开发者和管理员可从中获得最优的方法、务实的建议和实例代码来帮助他们掌握创建和维护企业级关系数据库所需的复杂技术。该书获得资深专家关于创建和维护健壮数据库的高屋建瓴般的视野和入木三分的剖析,十分适合有一定数据库基础的读者学习。 内容简介 本书是Inside Microsoft SQL Server 2000的作者Kalen Delaney的又一经典著作,是Inside Microsoft SQL Server 2005系列四本著作中的一本。本书对SQL Server 2005存储引擎方面的知识进行了全面而详细的阐述,包括数据库文件、日志和恢复、表、索引及其管理、锁定和并发等内容。除了解释设计理念与运作原理外,书中还辅之以大量简短而有力的实例。您将跟随一位广受欢迎的作家同时也是SQL Server资深专家一起深入探索SQL Server存储引擎的技术内幕。   本书适合于专业数据库开发者、BI开发者、DBA和以SQL Server作为后台数据库的一般应用程序开发者。本书不仅适合SQL Server 2005的初级读者,也适合SQL Server 2005的中高级读者。读者可以从中获得最优的方法、务实的建议和实例代码来帮助他们掌握创建和维护企业级关系数据库所需的复杂技术。本书是所有SQL Server 2005用户的案头必备之书。 作者简介 Kalen Delaney,她还是微软出版社inside SQL Sever丛书的编辑。她从1987年开始便一直从事SQL Server相关的工作,1995年被评为MVP(微软最有价值专家》。她同时也是Solid Quality Learning的首席顾问和创始人。除此之外,她还是SQL Server Magazine的优秀编辑和专栏作家,她还写作了大量的SQL Server类书籍,包括著名的Inside Microsoft SQL Server2000。 目录 前言 致谢 引言 第1章 SQL Server 2005 的安装与升级  1.1 SQL Server 2005安装前提   SQL Server 2005 版本   软件要求   硬件要求  1.2 安装前决策   安全性和用户上下文   字符与排序规则   排序次序   安装SQL Server的多个实例   安装SQL Server命名实例  1.3 做好安装准备   SQL Server 2005升级向导  1.4 迁移还是升级   迁移   升级   升级后的操作  1.5 选择组件   SQL Server数据库服务(数据库引擎)   Analysis Services   Reporting Services   Notification Services   Integration Services   工作站组件、联机丛书及开发工具  1.6 小结 第2章 SQL Server 2005体系结构  2.1 SQL Server引擎组件   观测数据库引擎行为   协议   表格格式数据流(TDS)端点   关系引擎   存储引擎   SQLOS  2.2 内存   缓冲池和高速数据缓冲区   访问内存中的数据页   管理数据高速缓冲区中的页面   检查点   管理其他高速缓存中的内存   调节内存大小   调节缓存池大小  2.3 小结 第3章 SQL Server 2005的配置  3.1 使用SQL Server 配置管理器   配置网络协议   默认的网络配置   管理服务  3.2 系统配置   任务管理   资源分配   系统分页文件的位置   非必需的服务   网络协议   与SQL Server 早期版本之间的兼容性   跟踪标记(Trace Flags)   SQL Server 的配置设定   内存选项   调度选项(Scheduling Options)   磁盘I/O 选项   查询处理选项   默认跟踪(Default Trace)  3.3 小结 第4章 数据库和数据库文件 第5章 日志和恢复 第6章 表 第7章 索引的内部构造和管理 第8章 锁定和并发

22,300

社区成员

发帖
与我相关
我的任务
社区描述
MS-SQL Server 疑难问题
社区管理员
  • 疑难问题社区
  • 尘觉
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧