数据库死锁

SQL Server数据库发生死锁时不会象Oracle那样自动生成一个跟踪文件.

有时可以在[管理]->[当前活动] 里看到阻塞信息(有时SQL Server企业管理器会因为锁太多而没有响应).

设定跟踪1204:

 

  
  USE MASTER
DBCC TRACEON (1204,-1)

显示当前启用的所有跟踪标记的状态:

 

  
  DBCC TRACESTATUS(-1)
取消跟踪1204:
DBCC TRACEOFF (1204,-1)

在设定跟踪1204后,会在数据库的日志文件里显示SQL Server数据库死锁时一些信息,

但那些信息很难看懂,需要对照SQL Server联机丛书仔细来看.

根据PAG锁要找到相关数据库表的方法:

 

  
  DBCC TRACEON (3604) 
DBCC PAGE (db_id,file_id,page_no)
DBCC TRACEOFF (3604)

请参考sqlservercentral.com上更详细的讲解.

从CSDN学到了一个找到死锁原因的方法.

我稍加改, 去掉了游标操作并增加了一些提示信息,写了一个系统存储过程sp_who_lock.sql.

需要的时候直接调用:

sp_who_lock

就可以查出引起死锁的进程和SQL语句.

SQL Server自带的系统存储过程sp_who和sp_lock也可以用来查找阻塞和死锁, 但没有这里介绍的方法好用.

如果想知道其它tracenum参数的含义,请看www.sqlservercentral.com文章

我们还可以设置锁的超时时间(单位是毫秒), 来缩短死锁可能影响的时间范围:

例如:

 

  
  use master
seelct @@lock_timeout
set lock_timeout 900000
-- 15分钟
seelct @@lock_timeout

 ==========以下为过程

use master
go
if exists (select * from dbo.sysobjects where id = object_id(N'[dbo].[sp_who_lock]') and OBJECTPROPERTY(id, N'IsProcedure') = 1)
drop procedure [dbo].[sp_who_lock]
GO
/***************************************************************************
//  创建 : fengyu  邮件 : maggiefengyu@tom.com  日期 :2004-04-30
//  修改 : 从http://www.csdn.net/develop/Read_Article.asp?id=26566学习到并改写 
//  说明 : 查看数据库里阻塞和死锁情况
***************************************************************************/

create procedure sp_who_lock
as
begin
declare @spid int,@bl int,
  @intTransactionCountOnEntry   int,
        @intRowcount       int,
        @intCountProperties     int,
        @intCounter       int

  create table #tmp_lock_who (
  id int identity(1,1),
  spid smallint,
  bl smallint)
 
  IF @@ERROR<>0 RETURN @@ERROR
 
  insert into #tmp_lock_who(spid,bl) select  0 ,blocked
    from (select * from sysprocesses where  blocked>0 ) a
    where not exists(select * from (select * from sysprocesses where  blocked>0 ) b
    where a.blocked=spid)
    union select spid,blocked from sysprocesses where  blocked>0

  IF @@ERROR<>0 RETURN @@ERROR 
  
-- 找到临时表的记录数
  select   @intCountProperties = Count(*),@intCounter = 1
  from #tmp_lock_who
 
  IF @@ERROR<>0 RETURN @@ERROR 
 
  if  @intCountProperties=0
    select '现在没有阻塞和死锁信息' as message

-- 循环开始
while @intCounter <= @intCountProperties
begin
-- 取第一条记录
    select   @spid = spid,@bl = bl
    from #tmp_lock_who where Id = @intCounter
  begin
   if @spid =0
            select '引起数据库死锁的是: '+ CAST(@bl AS VARCHAR(10)) + '进程号,其执行的SQL语法如下'
else
            select '进程号SPID:'+ CAST(@spid AS VARCHAR(10))+ '被' + '进程号SPID:'+ CAST(@bl AS VARCHAR(10)) +'阻塞,其当前进程执行的SQL语法如下'
DBCC INPUTBUFFER (@bl )
  end 

-- 循环指针下移
  set @intCounter = @intCounter + 1
end


drop table #tmp_lock_who

return 0
end

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Oracle数据库死锁是指两个或多个事务相互等待对方释放所持有的资源,导致事务无法继续执行,并且无法通过自动机制解决该冲突。当发生死锁时,数据库会检测到该情况,并选择一个事务进行回滚,以解除死锁。 出现死锁的原因通常是事务并发执行时对资源的竞争导致的,例如不同的事务同时请求相同的资源但按不同的顺序获取,或者事务持有了某些资源并请求其他事务所持有的资源。 为了处理Oracle数据库中的死锁问题,可以采取以下几种方法: 1. 检测死锁:Oracle数据库提供了系统视图和命令来检测死锁,例如使用`DBA_DDL_LOCKS`视图或`SELECT * FROM V$LOCKED_OBJECT`命令来查看被锁定的对象。 2. 解决死锁:一旦检测到死锁,Oracle数据库会自动选择一个事务进行回滚以解除死锁。这个过程是自动的,可以通过配置数据库参数来调整死锁检测和解除的行为。 3. 优化事务设计:设计良好的事务可以减少死锁的发生。例如,可以按照固定的顺序获取资源,或者减少事务之间对同一资源的竞争。 4. 调整数据库参数:可以通过调整Oracle数据库的参数来减少死锁的风险。例如,通过调整`DML_LOCKS`参数来控制事务对数据行的锁定方式。 5. 使用锁超时机制:可以在事务等待资源时设置锁超时,如果超过一定时间仍未获取到资源,则自动回滚事务。这可以减少死锁持续时间,但也可能导致事务执行失败。 以上是一些处理Oracle数据库死锁问题的常见方法,具体应根据实际情况选择合适的解决方案。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值