SQLSERVER排查阻塞
SQLServer常用性能分析语句
SQLServer常⽤性能分析语句--查看死锁情况SELECTDISTINCT'进程ID'=STR(a.spid, 4), '进程ID状态'=CONVERT(CHAR(10), a.status), '死锁进程ID'=STR(a.blocked, 2), '⼯作站名称'=CONVERT(CHAR(10), a.hostname), '执⾏命令的⽤户'=CONVERT(CHAR(10), SUSER_NAME(a.uid)), '数据库名'=CONVERT(CHAR(10), DB_NAME(a.dbid)), '应⽤程序名'=CONVERT(CHAR(10), a.program_name), '正在执⾏的命令'=CONVERT(CHAR(16), a.cmd), '登录名'= a.loginame, '执⾏语句'= b.textFROM master..sysprocesses a CROSS APPLYsys.dm_exec_sql_text(a.sql_handle) bWHERE a.blocked IN ( SELECT blockedFROM master..sysprocesses )-- and blocked <> 0ORDERBY STR(spid, 4)--查连接住信息(spid:57、58)select connect_time,last_read,last_write,most_recent_sql_handlefrom sys.dm_exec_connections where session_id in(57,58)--查看会话信息select login_time,host_name,program_name,login_name,last_request_start_time,last_request_end_timefrom sys.dm_exec_sessions where session_id in(57,58)--查看阻塞正在执⾏的请求selectsession_id,blocking_session_id,wait_type,wait_time,wait_resourcefromsys.dm_exec_requestswhereblocking_session_id>0--正在阻塞请求的会话的 ID。
SQLServer日常维护--查询当前正在执行的语句、死锁、堵塞
SQLServer⽇常维护--查询当前正在执⾏的语句、死锁、堵塞查询当前正在执⾏的语句:1SELECT2 der.[session_id],der.[blocking_session_id],3 stwaittype,sp.hostname,sp.program_name,sp.loginame,4 der.[start_time]AS'开始时间',5 der.[status]AS'状态',6 dest.[text]AS'sql语句',7DB_NAME(der.[database_id]) AS'数据库名',8 der.[wait_type]AS'等待资源类型',9 der.[wait_time]AS'等待时间',10 der.[wait_resource]AS'等待的资源',11 der.[logical_reads]AS'逻辑读次数'12FROM sys.[dm_exec_requests]AS der13INNER JOIN master.dbo.sysprocesses AS sp ON der.session_id=sp.spid14CROSS APPLY sys.[dm_exec_sql_text](der.[sql_handle]) AS dest15--WHERE [session_id]>50 AND session_id<>@@SPID16ORDER BY der.[session_id]17GOView Code是否堵塞:1SELECT spid,blocked,waittime,waittype,waitresource,p.dbid,cpu,physical_io,memusage,open_tran2 ,status,login_time,last_batch,hostname,program_name,hostprocess,loginame,cmd,text3FROM master.dbo.sysprocesses p CROSS APPLY sys.dm_exec_sql_text(p.sql_handle) s4WHERE blocked >0OR spid IN(SELECT blocked FROM master.dbo.sysprocesses WHERE blocked >0)5goView Code是否有未提交事物:1select spid,blocked,waittime,waittype,waitresource,p.dbid,cpu,physical_io,memusage,open_tran2 ,status,login_time,last_batch,hostname,program_name,hostprocess,loginame,cmd,text3from master.dbo.sysprocesses p cross apply sys.dm_exec_sql_text(p.sql_handle) s4where open_tran <>05goView Code死锁:1--drop table #deadlock23CREATE TABLE #deadlock(LogDate DATETIME,ProcessInfo VARCHAR(20),Text VARCHAR(2000))4go5INSERT INTO #deadlock EXEC xp_readerrorlog 0,1,'deadlock victim',NULL,'2016-07-20 14:00:00','2016-07-30','DESC'6goView Code死锁加nolock优化。
sqlserver trace 慢 排查思路-概述说明以及解释
sqlserver trace 慢排查思路-概述说明以及解释1.引言1.1 概述SQL Server Trace是SQL Server提供的一种用于监控和分析数据库性能的工具。
通过在数据库中设置Trace,可以捕获数据库操作的相关信息,如SQL语句执行时间、执行计划、IO操作等,从而帮助开发人员和数据库管理员找出潜在的性能问题。
在实际应用中,经常会遇到SQL Server执行查询变慢的情况,这可能是由于慢查询引起的。
慢查询会严重影响数据库的性能,并导致用户体验下降,因此及时排查慢查询是非常重要的。
本文将介绍SQL Server Trace的慢查询排查思路,帮助读者了解如何通过Trace分析工具来定位和解决SQL查询慢的问题。
通过本文的学习,读者将能够提升数据库性能,改善用户体验,为数据库的健康运行提供保障。
1.2文章结构1.2 文章结构本文将首先介绍SQL Server Trace的基本概念和作用,让读者对慢查询排查有一个全面的了解。
接着,将详细解释为什么排查慢查询是必要的,以及它对系统性能和用户体验的影响。
然后,我们将深入探讨SQL Server Trace慢查询排查的思路,包括具体的步骤和方法。
在结尾部分,将总结慢查询排查的关键步骤,并提出优化慢查询的建议,帮助读者更好地解决慢查询问题。
最后,通过一个简短的结语,为整篇文章画上一个完美的句号。
1.3 目的在SQL Server数据库管理中,我们经常会遇到慢查询的情况,这会导致系统性能下降,用户体验不佳,甚至影响业务运行。
因此,本文旨在探讨SQL Server Trace慢查询的排查思路,帮助读者了解如何通过追踪和分析数据库执行的SQL语句来排查慢查询的原因。
通过深入理解慢查询的产生原因,我们可以更好地优化SQL语句和数据库结构,提升系统性能,提高用户体验,确保业务的正常运行。
因此,本文的目的是帮助读者掌握慢查询排查的关键步骤,提供优化慢查询的建议,从而更好地管理和维护SQL Server数据库。
SQLSERVER数据库锁表的分析与解决
SQLSERVER数据库锁表的分析与解决数据库锁表的分析与解决上面介绍了内存溢出的原因和处理方法,下面再介绍一下数据库锁表及阻塞的原因和处理办法。
数据库和操作系统一样,是一个多用户使用的共享资源。
当多个用户并发地存取数据时,在数据库中就会产生多个事务同时存取同一数据的情况。
若对并发操作不加控制就可能会读取和存储不正确的数据,破坏数据库的一致性。
加锁是实现数据库并发控制的一个非常重要的技术。
在实际应用中经常会遇到的与锁相关的异常情况,当两个事务需要一组有冲突的锁,而不能将事务继续下去的话,就会出现死锁,严重影响应用的正常执行。
在数据库中有两种基本的锁类型:排它锁(Exclusive Locks,即X锁)和共享锁(Share Locks,即S 锁)。
当数据对象被加上排它锁时,其他的事务不能对它读取和修改。
加了共享锁的数据对象可以被其他事务读取,但不能修改。
数据库利用这两种基本的锁类型来对数据库的事务进行并发控制。
死锁的第一种情况一个用户A 访问表A(锁住了表A),然后又访问表B;另一个用户B 访问表B(锁住了表B),然后企图访问表A;这时用户A由于用户B已经锁住表B,它必须等待用户B释放表B才能继续,同样用户B要等用户A释放表A才能继续,这就死锁就产生了。
解决方法:这种死锁比较常见,是由于程序的BUG产生的,除了调整的程序的逻辑没有其它的办法。
仔细分析程序的逻辑,对于数据库的多表操作时,尽量按照相同的顺序进行处理,尽量避免同时锁定两个资源,如操作A和B两张表时,总是按先A后B的顺序处理,必须同时锁定两个资源时,要保证在任何时刻都应该按照相同的顺序来锁定资源。
死锁的第二种情况用户A查询一条纪录,然后修改该条纪录;这时用户B修改该条纪录,这时用户A的事务里锁的性质由查询的共享锁企图上升到独占锁,而用户B里的独占锁由于A有共享锁存在所以必须等A释放掉共享锁,而A由于B的独占锁而无法上升的独占锁也就不可能释放共享锁,于是出现了死锁。
SQL SERVER 阻塞与死锁
LOGO
检测阻塞
3. Other sys.dm_exec_connections(阻塞链中进程相关联接 信息) ----可了解联接中最后读写的时间及SQL语句
select * from sys.dm_exec_connections CROSS APPLY sys.dm_exec_sql_text(most_recent_sql_handle)as ST where session_id IN(55,56);
排他(X)
意向 架构 大容量更新(BU) 键同时获取一个资源上的锁。兼容时才 能获得。
请求模式 意向共享 共享 更新 意向排他 排他 现授予模式 IS S 是 是 是 是 否 是 是 是 否 否 否 U 是 是 否 否 否 否 IX 是 否 否 是 否 否 SIX 是 否 否 否 否 否 X 否 否 否 否 否 否
LOGO
可锁定资源
资源 RID KEY PAGE EXTENT HoBT TABLE FILE APPLICATION METADATA ALLOCATION_UNIT DATABASE 说明 堆中的单个行的行标识符 索引中用于保护序列化事务中的 键范围的行锁 数据库中的页,数据页或索引页 连续的8页 堆或B树。用于保护没有聚集索引 的表中的B树(索引)或堆数据页 整个表 数据库文件 应用程序专业资源 元数据 分配单元 数据库
sys.dm_exec_sessions (查询阻塞中涉及的会话) sys.dm_exec_requests (活动请求信息)
LOGO
终止和避免阻塞 1. Kill<spid> 2. SET LOCK_TIMEOUT timeout_period
LOGO
SqlServer阻塞的常见原因和解决办法
SqlServer阻塞的常见原因和解决办法阻塞的常见原因和解决办法:1. 由于语句运⾏时间太长⽽导致的阻塞,语句本⾝在正常运⾏中,只须等待某些系统资源解决办法:a. 语句本⾝有没有可优化的空间b. Sql Server 整体性能如何,是不是有资源瓶颈影响了语句执⾏速度,如内存、硬盘和 CPU 等2. 由于⼀个未按预期提交的事务导致的阻塞这⼀类阻塞的特征,就是问题连接早就进⼊了空闲状态(sysprocesses.status='sleeping'和sysprocesses.cms='awaiting command'),但是,如果检查 sysprocesses.open_tran,就会发现它不为0,以及事务没有提交。
这类问题很多都是因为应⽤端遇到了⼀个执⾏超时,或者其他原因,当时执⾏的语句倍提前终⽌了,但是连接还保留着。
应⽤没有跟随发来的事务提交或回滚指令,导致⼀个事务被遗留在 Sql Server ⾥。
解决办法:应⽤程序本⾝必须意识到任何语句都有可能遇到意外终⽌的情况,做好错误处理⼯作。
这些⼯作包括:· 在做 Sql Server 调⽤的时候,须加上错误捕捉和处理语句:If @@Trancount>0 RollBack Tran;(在程序中设置If @@Error<>0 Rollback Tran; 并不总是能执⾏到该语句)· 设置连接属性"Set XACT_ABORT ON"。
如果没有办法很规范应⽤程序的错误扑捉和处理语句,⼀个最快的⽅法就是在每个连接建⽴以后,或是容易出问题的存储过程开头,运⾏ "Set XACT_ABORT ON"·考虑是否要关闭连接池。
发⼀句 sp_reset_connection 命令清理当前连接上次遗留下来的所有对象,包括回滚未提交的事务。
3. 由于客户端没有及时把结果集取出⽽导致的语句长时间运⾏语句在 Sql Server 内执⾏总时间不仅包含 Sql Server 的执⾏时间,还包含把结果集发给客户端的时间。
sqlserverexception read timed out -回复
sqlserverexception read timed out -回复主题:深入了解SQL Server异常读取超时错误([SQLServerException Read Timed Out])引言:在使用SQL Server时,我们经常会遇到各种异常。
其中之一是read timed out异常,也就是读取超时错误。
这个错误通常在执行数据库查询操作时出现,它表示服务器在执行查询时花费的时间超过了预定的超时时间。
本文将带领读者深入了解SQL Server异常读取超时错误的原因,如何诊断和解决这个问题。
第一部分:读取超时错误的原因1. 网络问题:读取超时错误可能是由于网络问题导致的。
例如,网络连接不稳定、网络带宽受限等问题会导致数据传输的延迟,从而触发超时错误。
2. 查询复杂度:如果查询涉及大量数据或执行复杂的计算操作,可能会导致查询时间超过了预定的超时时间。
3. 锁定和阻塞:如果其他事务正在进行长时间的读或写操作,并且正在占用查询所需的资源,就可能导致查询超时。
4. 服务器配置:如果SQL Server的配置未经过优化,例如内存不足、硬件资源利用率低等,也会导致读取操作的超时错误。
第二部分:诊断读取超时错误1. 检查网络连接:首先,我们需要检查客户端和服务器之间的网络连接是否稳定。
可以使用ping命令来测试网络延迟或使用网络监控工具来检查网络带宽。
2. 查看查询执行计划:使用SQL Server Management Studio等工具查看查询的执行计划。
执行计划可以告诉我们查询所需的资源和操作的顺序,帮助我们找出可能导致超时的瓶颈。
3. 检查锁定和阻塞情况:使用sys.dm_tran_locks视图或SQL Server Profiler工具来检查当前正在进行的事务和锁定情况。
如果有其他事务正在占用所需的资源,我们可以考虑优化查询或调整事务隔离级别来避免超时错误。
4. 检查服务器配置:查看SQL Server的配置参数,如最大内存限制、最大并发连接数等。
sqlserver无法用排他锁锁定该数据库,以执行该操作。(MicrosoftSQLSer。。。
ቤተ መጻሕፍቲ ባይዱ请求出错错误代码503请尝试刷新页面重试
sqlserver无 法 用 排 他 锁 锁 定 该 数 据 库 , 以 执 行 该 操 作 。 ( MicrosoftSQLSer。。。
原因是因为其他的线程占用这SqlServer的连接 可通过SQL查看
select spid from master.dbo.sysprocesses where dbid=db_id('databaseName')
采用SQL杀死占用方
kill spid
记得在执行这个命令的时候切换为别的数据库,要不然本身会占用一个连接,还杀不掉 如果杀掉后,不停的生成新的SPID,就把线上可能用到这个数据库的账户权限暂时取消掉,这样就可以避免线上的环境再连接到这个数据 库
使用下面命令重命名
use master go exec sp_renamedb @dbname='KCMainDB', @newname='KCMainDB-test'; go
关于软件连接不上SQLSERVER数据库的排查方法
关于软件连接不上SQLSERVER数据库的排查方法当软件无法连接SQL Server数据库时,可以采取以下方法进行排查: 1.检查网络连接:-确保客户端和服务器在同一网络中,并且网络连接正常。
-检查防火墙设置,确保客户端可以访问数据库服务器的端口(默认为1433)。
- 使用 ping 命令测试客户端是否可以访问数据库服务器。
2. 检查SQL Server服务是否运行:- 打开SQL Server Configuration Manager,检查SQL Server服务是否已启动。
- 检查SQL Server的错误日志,查看是否有启动错误或其他错误信息。
3. 检查SQL Server实例名称和用户名/密码:- 确保软件连接字符串中指定的SQL Server实例名称正确。
-检查连接字符串中的用户名和密码是否正确,确保具有访问数据库的权限。
4.检查数据库实例是否允许远程连接:- 在SQL Server Management Studio中,找到数据库实例,右键单击选择“属性”。
-在“连接”选项卡中,确保“允许远程连接到此服务器”选项已启用。
5.检查数据库的权限和登录:-确保登录用户具有足够的权限来访问所需的数据库。
- 使用SQL Server Management Studio登录到数据库,检查登录用户的权限和角色。
6. 检查SQL Server配置参数:- 检查SQL Server配置参数,如最大连接数和最大内存限制。
-可能需要调整这些参数以提高连接性能。
7. 检查SQL Server错误日志:- 在SQL Server Management Studio中,打开“管理”节点,然后展开“SQL Server错误日志”。
-检查日志寻找与连接问题相关的错误消息。
根据错误消息进行相应的调整。
8.检查网络传输协议:- 在SQL Server Configuration Manager中,选择SQL Server网络配置。
SqlServer异常处理常用步骤
SqlServer异常处理常⽤步骤SQL Server常见的问题主要是SQL问题造成,常见的主要是CPU过⾼和阻塞。
⼀、CPU过⾼的问题1、查询系统动态视图查询执⾏时间长的sql语句WITH ProcessCTE(blocked) AS(SELECT spid FROM sys.sysprocesses WHERE cpu>500)SELECT distinct a.*FROM (SELECT TEXT,AA.* FROM sys.sysprocesses AACROSS APPLY sys.dm_exec_sql_text(AA.sql_handle)) aJOIN ProcessCTE bucte WITH(NOLOCK)ON bucte.blocked=a.spid--where loginame = 'TCScenery'ORDER BY a.CPU⼆、阻塞问题1、查询系统动态视图查询阻塞的sql语句WITH ProcessCTE(blocked) AS(SELECT blocked FROM sys.sysprocesses WHERE blocked>0unionSELECT blocked FROM sys.sysprocesses WHERE blocked>0)SELECT distinct a.*FROM (SELECT TEXT,AA.* FROM sys.sysprocesses AACROSS APPLY sys.dm_exec_sql_text(AA.sql_handle)) aJOIN ProcessCTE bucte WITH(NOLOCK)ON bucte.blocked=a.spidORDER BY a.blocked2、使⽤系统⾃带的存储过程Sp_who2和sp_lock以及使⽤dbcc inputbuffer(spid) 也可以⽤来分析阻塞sp_who可以返回如下信息: (可选参数LoginName, 或active代表活动会话数)Spid (系统进程ID)status (进程状态)loginame (⽤户登录名)hostname(⽤户主机名)blk (阻塞进程的SPID)dbname (进程正在使⽤的数据库名)Cmd (当前正在执⾏的命令类型)sp_who2除了显⽰上⾯sp_who的输出信息外,还显⽰下⾯的信息: (可选参数LoginName, 或active代表活动会话数) CPUTime (进程占⽤的总CPU时间)DiskIO (进程对磁盘读的总次数)LastBatch (客户最后⼀次调⽤存储过程或者执⾏查询的时间)ProgramName (⽤来初始化连接的应⽤程序名称,或者主机名)下⾯是sp_who的⽤法,sp_who2与此类似A.列出全部当前进程以下⽰例使⽤没有参数的 sp_who 来报告所有当前⽤户。
SQLServer死锁和阻塞的源头查询
FETCH NEXT FROM s_cur INTO @spid,@bl
end
CLOSE s_cur
DEALLOCATE s_cur
exec sp_who2
-----------------方法二
USE MASTER
GO
DECLARE @spid INT
--查询出死锁的SPID
SELECT @spid=blocked
FROM (SELECT * FROM sysprocesses WHERE blocked > 0) a
WHERE NOT EXISTS(SELECT * FROM (SELECT *E blocked > 0) b
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语法如下'
WHERE a.blocked=@spid)
--输出引起死锁的操作
DBCC INPUTBUFFER (@spid)
--KILL引起死锁的进程
EXEC ('KILL ' + @spid)
SqlServer查询和Kill进程死锁的语句
SqlServer查询和Kill进程死锁的语句查询死锁进程语句selectrequest_session_id spid,OBJECT_NAME(resource_associated_entity_id) tableNamefromsys.dm_tran_lockswhereresource_type='OBJECT'杀死死锁进程语句kill spid下⾯再给⼤家分享⼀段关于sqlserver检测死锁;杀死锁和进程;查看锁信息--检测死锁--如果发⽣死锁了,我们怎么去检测具体发⽣死锁的是哪条SQL语句或存储过程?--这时我们可以使⽤以下存储过程来检测,就可以查出引起死锁的进程和SQL语句。
SQL Server⾃带的系统存储过程sp_who和sp_lock也可以⽤来查找阻塞和死锁, 但没有这⾥介绍的⽅法好⽤。
use mastergocreate procedure sp_who_lockasbegindeclare @spid int,@bl int,@intTransactionCountOnEntry int,@intRowcount int,@intCountProperties int,@intCounter intcreate table #tmp_lock_who (id int identity(1,1),spid smallint,bl smallint)IF @@ERROR<>0 RETURN @@ERRORinsert into #tmp_lock_who(spid,bl) select 0 ,blockedfrom (select * from sysprocesses where blocked>0 ) awhere not exists(select * from (select * from sysprocesses where blocked>0 ) bwhere a.blocked=spid)union select spid,blocked from sysprocesses where blocked>0IF @@ERROR<>0 RETURN @@ERROR-- 找到临时表的记录数select @intCountProperties = Count(*),@intCounter = 1from #tmp_lock_whoIF @@ERROR<>0 RETURN @@ERRORif @intCountProperties=0select '现在没有阻塞和死锁信息' as message-- 循环开始while @intCounter <= @intCountPropertiesbegin-- 取第⼀条记录select @spid = spid,@bl = blfrom #tmp_lock_who where Id = @intCounterbeginif @spid =0select '引起数据库死锁的是: '+ CAST(@bl AS VARCHAR(10)) + '进程号,其执⾏的SQL语法如下'elseselect '进程号SPID:'+ CAST(@spid AS VARCHAR(10))+ '被' + '进程号SPID:'+ CAST(@bl AS VARCHAR(10)) +'阻塞,其当前进程执⾏的SQL语法如下'DBCC INPUTBUFFER (@bl )end-- 循环指针下移set @intCounter = @intCounter + 1enddrop table #tmp_lock_whoreturn 0end--杀死锁和进程--如何去⼿动的杀死进程和锁?最简单的办法,重新启动服务。
SQLserver阻塞
SQL server阻塞(来自微软技术支持人员)作者:lyf1840阻塞定义===============当来自应用程序的第一个连接控制锁而第二个连接需要相冲突的锁类型时,将发生阻塞。
其结果是强制第二个连接等待,而在第一个连接上阻塞。
不管是来自同一应用程序还是另外一台客户机上单独的应用程序,一个连接都可以阻塞另一个连接。
说明一些需要锁保护的操作可能不明显,例如系统目录表和索引上的锁。
大多数阻塞问题的发生是因为一个进程控制锁的时间过长,导致阻塞的进程链都在其它进程上等待锁。
常见的阻塞情形包括===============1 .提交执行时间长的查询。
长时间运行的查询会阻塞其它查询。
例如,影响很多行的 DELETE 或 UPDATE操作能获取很多锁,这些锁不论是否升级到表锁都阻塞其它查询。
因此,一般不要将长时间运行的决策支持查询和联机事务处理 (OLTP)查询混在一起。
解决方案是想办法优化查询,如更改索引、将大的复杂查询分成简单的查询或在空闲时间或单独的计算机上运行查询。
2 .查询不适当地使用游标。
游标可能是在结果集中浏览的便利方法,但使用游标可能比使用面向集合的查询慢。
3 .取消没有提交或回滚的查询。
如果应用程序取消查询(如使用开放式数据库连接 (ODBC) sqlcancel 函数)但没有同时发出所需数目的ROLLBACK 和 COMMIT语句,则会发生这种情况。
取消查询并不自动回滚或提交事务。
取消查询后,所有在事务获取的锁都将保留。
应用程序必须提交或回滚已取消的事务,从而正确地管理事务嵌套级。
4 .应用程序没处理完所有结果。
将查询发送到服务器后,所有应用程序必须立即完成提取所有结果行。
如果应用程序没有提取所有结果行,锁可能会留在表上而阻塞其他用户。
如果使用的应用程序将Transact-SQL 语句透明地提交给服务器,则该应用程序必须提取所有结果行。
如果应用程序没这样做(如果无法配置它执行此操作),则可能无法解决阻塞问题。
SQLServer死锁产生原因及解决办法.
SQLServer死锁产⽣原因及解决办法.其实所有的死锁最深层的原因就是⼀个:资源竞争表现⼀: ⼀个⽤户A 访问表A(锁住了表A),然后⼜访问表B,另⼀个⽤户B 访问表B(锁住了表B),然后企图访问表A,这时⽤户A由于⽤户B已经锁住表B,它必须等待⽤户B释放表B,才能继续,好了他⽼⼈家就只好⽼⽼实实在这等了,同样⽤户B要等⽤户A释放表A才能继续这就死锁了。
解决⽅法: 这种死锁是由于你的程序的BUG产⽣的,除了调整你的程序的逻辑别⽆他法 仔细分析你程序的逻辑: 1:尽量避免同时锁定两个资源 2: 必须同时锁定两个资源时,要保证在任何时刻都应该按照相同的顺序来锁定资源.表现⼆: ⽤户A读⼀条纪录,然后修改该条纪录。
这是⽤户B修改该条纪录,这⾥⽤户A的事务⾥锁的性质由共享锁企图上升到独占锁(for update),⽽⽤户B⾥的独占锁由于A有共享锁存在所以必须等A释放掉共享锁,⽽A由于B的独占锁⽽⽆法上升的独占锁也就不可能释放共享锁,于是出现了死锁。
这种死锁⽐较隐蔽,但其实在稍⼤点的项⽬中经常发⽣。
解决⽅法: 让⽤户A的事务(即先读后写类型的操作),在select 时就是⽤Update lock 语法如下:select * from table1 with(updlock) where ....--------------------------------------------------------------------------------接上⾯⽂章,继续探讨数据库死锁问题死锁,简⽽⾔之,两个或者多个trans,同时请求对⽅正在请求的某个对象,导致双⽅互相等待。
简单的例⼦如下:trans1 trans2------------------------------------------------------------------------------------------------1.IDBConnection.BeginTransaction 1.IDBConnection.BeginTransaction2.update table A 2.update table B3.update table B 3.update table Amit mit那么,很容易看到,如果trans1和trans2,分别到达了step3,那么trans1会请求对于B的X锁,trans2会请求对于A的X锁,⽽⼆者的锁在step2上已经被对⽅分别持有了。
SQLServer查询、搜索命令、语句
SQLServer查询、搜索命令、语句SQL Server⾃带的系统存储过程sp_who和sp_lock也可以⽤来查找阻塞和死锁⼀. 阻塞查询 sp_lock执⾏ exec sp_lock 下⾯列下关键字段spid 是指进程ID,这个过滤掉了系统进程,只展⽰了⽤户进程spid>50。
dbid 指当前实例下的哪个数据库 , 使⽤DB_NAME() 函数来标识数据库type 请求锁住的模式mode 锁的请求状态GRANT:已获取锁。
CNVRT:锁正在从另⼀种模式进⾏转换,但是转换被另⼀个持有锁(模式相冲突)的进程阻塞。
WAIT:锁被另⼀个持有锁(模式相冲突)的进程阻塞。
总结:当mode 不为GRANT状态时,需要了解当前锁的模式,以及通过进程ID查找当前sql 语句例如当前进程ID是416,且mode状态为WAIT 时,查看⽅式 DBCC INPUTBUFFER(416)⽤sp_lock查询显⽰的信息量很少,也很难看出谁被谁阻塞。
所以当数据库版本为2005及以上时不建议使⽤。
⼆.阻塞查询 dm_tran_locksSELECTt1.resource_type,t1.resource_database_id,t1.resource_associated_entity_id,t1.request_mode,t1.request_session_id,t2.blocking_session_idFROM sys.dm_tran_locks as t1INNER JOIN sys.dm_os_waiting_tasks as t2ON t1.lock_owner_address = t2.resource_address;上⾯查询只显⽰有阻塞的会话,关注blocking_session_id 也就是被阻塞的会话ID,同样使⽤DBCC INPUTBUFFER来查询sql语句三.阻塞查询 sys.sysprocessesSELECTspid,kpid,blocked,waittime AS'waitms',lastwaittype,DB_NAME(dbid)AS DB,waitresource,open_tran,hostname,[program_name],hostprocess,loginame,[status]FROM sys.sysprocesses WITH(NOLOCK)WHERE kpid>0AND[status]<>'sleeping'AND spid>50sys.sysprocesses 能显⽰会话进程有多少,等待时间, open_tran有多少事务,阻塞会话是多少. 整体内容更为详细。
怎样查出SQLServer的性能瓶颈
怎样查出SQLServer的性能瓶颈SQL Server是一款常用的关系型数据库管理系统,可以用于存储和管理大量的数据。
然而,在使用SQL Server时,我们常常会遇到性能瓶颈的问题,导致数据库操作变慢,影响系统的正常运行。
为了解决这些问题,我们需要对SQL Server进行性能优化,首先要查出性能瓶颈。
下面将详细介绍如何查出SQL Server的性能瓶颈。
第一步:监控系统性能要查出SQL Server的性能瓶颈,首先要对系统的性能进行监控。
可以使用SQL Server自带的性能监视工具,如Performance Monitor和SQL Server Profiler。
Performance Monitor可以监控系统的硬件性能,如CPU利用率、内存使用情况、磁盘IO等;SQL Server Profiler可以监控数据库的性能,如查询执行时间、锁定情况等。
第二步:识别慢查询在监控系统性能的基础上,我们还需要识别出哪些查询存在性能问题。
可以通过查询执行计划、系统视图和性能监视器等方式来判断哪些查询的执行时间较长或者占用较多的系统资源。
1. 使用查询执行计划:在SQL Server Management Studio中执行查询时,可以选择显示查询执行计划。
执行计划可以告诉我们查询的执行过程,包括使用了哪些索引、是否进行了表扫描等。
可以通过查看执行计划中的耗时最长的操作节点来判断性能瓶颈所在。
2. 使用系统视图:SQL Server中有一些系统视图,如sys.dm_exec_query_stats和sys.dm_exec_query_plan,可以查询有关查询的性能信息。
可以通过查找执行时间最长的查询语句,并分析其执行计划,判断性能瓶颈所在。
3. 使用性能监视器:可以通过性能监视器来监控数据库的性能指标,如平均响应时间、平均锁等待时间等。
可以根据这些指标判断哪些查询存在性能问题。
第三步:分析性能瓶颈在识别出慢查询之后,我们需要对慢查询进行分析,找出性能瓶颈所在。
sqlserver解除死锁的方法
sqlserver解除死锁的方法一、什么是死锁死锁是指两个或多个事务在执行过程中,因互相请求对方所占有的资源而陷入互相等待的状态,导致程序无法继续执行的情况。
在SQL Server中,当两个或多个事务同时请求同一资源时,如果这些资源被另一个事务所持有,则会发生死锁。
二、如何诊断死锁1. 查看SQL Server错误日志SQL Server错误日志中会记录死锁事件的详细信息。
可以通过查看错误日志来确定哪些连接发生了死锁。
2. 使用系统视图使用系统视图sys.dm_tran_locks和sys.dm_exec_sessions来查看当前活动连接和它们所持有的锁。
3. 使用SQL Profiler使用SQL Profiler来监视并捕获死锁事件。
在Profiler中可以选择Deadlock Graph事件来查看详细信息。
三、如何解除死锁1. 调整应用程序代码应用程序代码可能会导致死锁。
可以通过调整代码来避免死锁的发生,例如更改事务顺序或减少对共享资源的访问。
2. 调整数据库设计数据库设计可能会导致死锁。
可以通过调整数据库设计来避免死锁的发生,例如更改索引策略或分割表。
3. 提高事务隔离级别提高事务隔离级别可以减少死锁的发生。
但是需要注意,提高隔离级别会增加锁的竞争,可能导致性能下降。
4. 使用锁超时可以设置锁超时时间来避免死锁的发生。
当一个连接持有锁超过指定时间后,系统会自动释放该锁。
5. 手动解除死锁手动解除死锁是一种最后的手段。
可以使用以下命令手动解除死锁:SELECT * FROM sys.dm_tran_locks WHERE request_session_id IN(SELECT blocked_session_id FROM sys.dm_exec_requests WHERE status='suspended')KILL <session_id>以上命令会查找所有被阻塞的连接,并杀掉其中一个连接以解除死锁。
数据库系统异常排查之思路
数据库系统异常排查之思路数据库系统异常是DBA经常要⾯临的情景,⼀名有⼀定从业经验的DBA,都会有⾃⼰⼀套故障排查的⽅法和步骤,此⽂为为⼤家介绍⼀下通过系统性能视图(SQLServer05以上版本)来排查系统异常的基本⽅法,希望能对⼤家有所帮助。
这⾥分两部分来介绍:⼀. 从数据库连接情况来判断异常:1. ⾸先我们来看⼀下⽬前数据库系统所有请求情况:--request infoselect s.session_id, s.status,db_name(r.database_id) as database_name,s.login_name,s.login_time, s.host_name,c.client_net_address,c.client_tcp_port,s.program_name,r.cpu_time, r.reads, r.writes,c.num_reads,c.num_writes,s.client_interface_name,st_request_start_time, st_request_end_time,c.connect_time, _transport, _packet_size,r.start_time, r.status, mand,r.blocking_session_id, r.wait_type,r.wait_time, st_wait_type, r.wait_resource, r.open_transaction_count,r.percent_complete,r.granted_query_memoryfrom Sys.dm_exec_requests r with(nolock)right outer join Sys.dm_exec_sessions s with(nolock)on r.session_id = s.session_idright outer join Sys.dm_exec_connections c with(nolock)on s.session_id = c.session_idwhere s.session_id >50order by s.session_id某台⽣产机运⾏情况:这个查询将⽬前数据库中的所有请求都显⽰出来了,其中⽐较重要的有Status、Login_name、Host_Name,Client_Net_Address、Program_name等,但是信息⽐较多,我们很难查看有什么异常,但是可以通过⼀图中红⾊圈的数字:441 初步判断连接数是否超过了平时的标准(很多时候系统异常是连接数过多造成的,⽽连接数过多⼜是因为其他原因影响的)。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
SQL Server允许并发操作,BLOCKING是指在某一操作没有完成之前,其他操作必须等待,以便于保证数据的完整性。
BLOCKING的解决方法要查看BLOCKING的头是什么,为什么BLOCKING头上的语句执行的很慢。
通常来讲只要我们能找到BLOCKING头上的语句,我们总能够想出各种各种的办法,来提升性能,缓解或解决BLOCKING的问题。
但是问题的关键是,我们不知道BLOCKING什么时候会发生。
用户跟我们抱怨数据库性能很差,等我们连上数据库去查看的时候,那时候有可能BLOCKING可能就已经过去了。
性能又变好了。
或者由于问题的紧急性,我们直接重新启动服务器以恢复运营。
但是问题并没有最终解决,我们不知道下次问题会在什么时候发生。
BLOCKING问题的后果比较严重。
因为终端用户能直接体验到。
他们提交一个订单的时候,无论如何提交不上去,通常几秒之内能完成的一个订单提交,甚至要等待十几分钟,才能提交完成。
更有甚者,极严重的BLOCKING能导致SQL Server 停止工作。
如下面的SQL ERRORLOG所表示, 在短短的几分钟之内,SPID数据从158增长到694, 并马上导致SQL Server打了一个dump, 停止工作。
我们很容易推断出问题的原因是由于BLOCKING导致的,但是我们无法得知BLOCKING HEADER是什么,我们必须要等下次问题重现时,辅之以工具,才能得知BLOCKING HEADER在做什么事情。
如果信息抓取时机不对,我们可能要等问题发生好几次,才能抓到。
这时候,客户和经理就会有抱怨了。
因为我们的系统是生产系统,问题每发生一次,都会对客户带来损失。
2011-06-01 16:22:30.98 spid1931 Alert There are 158 Active database sessions which is too high.2011-06-01 16:23:31.16 spid3248 Alert There are 342 Active database sessions which is too high.2011-06-01 16:24:31.06 spid3884 Alert There are 517 Active database sessions which is too high.2011-06-01 16:25:31.08 spid3688 Alert There are 694 Active database sessions which is too high.2011-06-01 16:26:50.93 Server Using 'dbghelp.dll' version '4.0.5'2011-06-01 16:26:50.97 Server **Dump thread - spid = 0, EC = 0x00000000000000002011-06-01 16:26:50.97 Server ***Stack Dump being sent toD:\MSSQL10.INSTANCE\MSSQL\LOG\SQLDump0004.txt2011-06-01 16:26:50.97 Server ********************************************************************************2011-06-01 16:26:50.97 Server *2011-06-01 16:26:50.97 Server * BEGIN STACK DUMP:2011-06-01 16:26:50.97 Server * 06/01/11 16:26:50 spid 41242011-06-01 16:26:50.97 Server *2011-06-01 16:26:50.97 Server * Deadlocked Schedulers2011-06-01 16:26:50.97 Server *2011-06-01 16:26:50.97 Server ********************************************************************************2011-06-01 16:26:50.97 Server *-------------------------------------------------------------------------------2011-06-01 16:26:50.97 Server * Short Stack Dump2011-06-01 16:26:51.01 Server Stack Signature for the dump is 0x0000000000000258BLOCKING的信息抓取有很多种方法。
这里罗列了几种。
并且对每种分析它的优缺点。
以便我们选择。
在枚举方法之前,我们先简单演示一下BLOCKING.我们首先创建一个测试表:DROPTABLE [TESTTABLE]GOCREATETABLE [dbo].[TESTTABLE]([ID] [int] NULL,[NAME] [nvarchar](50)NULL)GOINSERTINTO TESTTABLE VALUES (1,'aaaa')GO然后打开一个查询窗口,执行下面的语句, 该语句修改一行数据,并等待3分钟,然后在结束transactionBEGINTRANSACTIONUPDATE TESTTABLE SET [NAME] ='bbbb'WHERE [ID] = 1WAITFORDELAY'00:03:00'COMMITTRANSACTION这时候,如果打开另外一个查询窗口,执行下面的语句,下面的语句就会被BLOCK 住。
UPDATE TESTTABLE SET [NAME] ='cccc'WHERE [ID] = 1方法一, 抓取SQL Profiler======================SQL Profiler里面包含大量的信息。
其中有一个事件在Errors andWarnings->Blocked Process Report专门用来获得blocking的情况。
但是因为信息量比较大,而且我们并不能很好的估算在什么时候会产生blocking,另外在生产环境使用Profiler, 对性能可能会有影响,所以SQL Profiler并不是最合适的工具。
我们在这里并不对它赘述。
方法二, 执行查询================如果我们检查问题的时候,blocking还存在,那么,我们可以直接可以运行几个查询,得知BLOCKING HEADER的信息SELECT*FROM sys.sysprocesses where spid>50上述查询只是告诉我们,BLOCKING HEADER的头是SPID=53, 但是并没有告诉我们SPID=53在做什么事情。
我们可以用下面的查询,得到SPID=53的信息DBCC INPUTBUFFER(53)我们可以把上述的两个查询合并起来,用下面的查询:SELECT SPID=p.spid,DBName =convert(CHAR(20),),ProgramName =program_name,LoginName =convert(CHAR(20),),HostName =convert(CHAR(20),hostname),Status= p.status,BlockedBy = p.blocked,LoginTime = login_time,QUERY =CAST(TEXTASVARCHAR(MAX))FROMMASTER.dbo.sysprocesses pINNERJOIN MASTER.dbo.sysdatabases dON p.dbid= d.dbidINNERJOIN MASTER.dbo.syslogins lON p.sid= l.sidCROSSAPPLY sys.dm_exec_sql_text(sql_handle)WHERE p.blocked = 0ANDEXISTS(SELECT 1FROMMASTER..sysprocesses p1WHERE p1.blocked = p.spid)这样,一次执行,就能告诉我们BLOCKING header的SPID信息,以及该SPID 在做的语句。
我们可以进一步研究该语句,以理解为什么该语句执行很慢。
用这个方法有一个缺点,就是使用的时候,要求BLOCKING是存在的。
如果BLOCKING已经消失了,那么我们的方法就不管用了。
方法三,长期执行一个BLOCKING SCRIPT==================================因为我们通常无法知道BLOCKING什么时候会产生,所以通常的办法是我们长期运行一个BLOCKING SCRIPT, 这样,等下次发生的时候,我们就会有足够的信息。
长期运行BLOCKING SCRIPT对性能基本上是没有影响的。
因为我们每隔10秒钟抓取一次信息。
缺点是,如果问题一个月才发生一次,那么,我们的BLOCKING 日志信息会很大。
所以这种方法适用于几天之内能重现问题。
运行方法如下:如果要停止运行,我们按ctrl+c就可以了。
BLOCKING的信息存在log.out这个文件中我们可以打开log.out这个文件, 会发现SPID 54被 SPID 53给Block住了。
而随后,我们可以看到SPID=53在做什么事情:下面是BLOCKING SCRIPT的脚本, 我们可以把它存为blocking.sqlusemastergowhile 1 =1beginprint'Start time: '+convert(varchar(26),getdate(), 121)Print'Running processes'select spid, blocked, waittype, waittime, lastwaittype, waitresource,dbid,uid, cpu, physical_io, memusage, login_time,last_batch,open_tran,status, hostname,program_name, cmd, net_library, loginame from sysprocesses--where (kpid <> 0 ) or (spid < 51)-- Change it if you only want to see the working processesprint'*********lockinfor***********'select convert(smallint, req_spid)As spid,rsc_dbid Asdbid,rsc_objid AsObjId,rsc_indid As IndId,substring(, 1, 4)AsType,substring(rsc_text, 1, 16)asResource,substring(, 1, 8)As Mode,substring(, 1, 5)AsStatusfrommaster.dbo.syslockinfo,master.dbo.spt_values v,master.dbo.spt_values x,master.dbo.spt_values uwheremaster.dbo.syslockinfo.rsc_type = v.numberand v.type='LR'and master.dbo.syslockinfo.req_status = x.numberand x.type='LS'and master.dbo.syslockinfo.req_mode + 1 = u.numberand u.type='L'orderby spidprint'inputbuffer for running processes'declare @spid varchar(6)declare ibuffer cursorfast_forwardforselect cast(spid asvarchar(6))as spid from sysprocesses where spid >50 open ibufferfetchnextfrom ibuffer into @spidwhile (@@fetch_status!=-1)beginprint''print'DBCC INPUTBUFFER FOR SPID '+ @spidexec ('dbcc inputbuffer ('+ @spid +')')fetchnextfrom ibuffer into @spidenddeallocate ibufferwaitfordelay'0:0:10'End这种方法的缺陷就是,log.out会比较巨大,会占用很大的空间,如果blocking 一个月甚至更长时间才发生一次,那我们的这个方法就不太适宜。