1. 什么是"阻塞"?

阻塞是SQL数据库应用"锁"机制的一个副作用。当一个应用请求针对某个数据库对象(例如全表,某行数据, 或者是某个数据页)加锁后,那么这个锁会阻塞其它的应用请求。这就好像你把家里的大门上了锁,你的老婆没有钥匙,只能等待你回家。她进入房间的请求被你阻塞了,不得不等待,直到你解锁开门。对于数据库来说短暂的阻塞是可以被接受的。而且短暂的阻塞也是数据库的常态。只有当阻塞的时间超过了人们的容忍时间,这种阻塞现象需要DBA的关注。

2. 如何发现"当前"的"阻塞"? 

通过下列工具可以发现当前正在发生的阻塞现象

  • 活动监视器 Activity Monitor
  • sp_who2
  • 动态性能视图(DMV)
  • 会话 ID: 是建立连接时分配给每个用户连接的唯一整数 (int)。
  • 等待时间(毫秒):此任务等待资源所用的时间(毫秒)。如果任务没有等待,则等待时间为 0。
  • 等待类型:最近或当前等待类型的名称。
  • 等待资源:所需资源的名称。
  • 阻塞者:如果有阻塞会话,则为正阻塞任务的会话的 ID。
  • 头阻塞程序:如果有阻塞会话,则标识导致第一个阻塞条件的会话。值为 1 表示其他会话的头阻塞程序。

 

SQL Server 2008 启用线程池 sqlserver线程阻塞_sql

 

2.2 sp_who2

在master数据库下,运行如下语句: 

1. exec sp_who2

 你会看到下面的信息,它类似活动监视器,显示当前用户请求的阻塞信息.只是表格化了,便于我们过滤一些不相关的内容.

SQL Server 2008 启用线程池 sqlserver线程阻塞_sql_02

 通过下面的语句可以将sp_who2的显示信息导入到一张临时表里.当然你也可以导入到一张永久表里.

1. CREATETABLE #sp_who2 (SPID INT,Status VARCHAR(255),  
2. VARCHAR(255),HostName  VARCHAR(255),  
3. VARCHAR(255),DBName  VARCHAR(255),  
4. VARCHAR(255),CPUTime INT,  
5. INT,LastBatch VARCHAR(255),  
6. VARCHAR(255),SPID2 INT,  
7. INT)  
8.   
9. INSERTINTO #sp_who2 EXECsp_who2  
10. SELECT  *   
11. FROM  #sp_who2  
12. WHERE       DBName <> 'master'  
13. ORDER BY    DBName ASC  
14.    
15. DROP TABLE #sp_who2

  2.3 使用动态性能视图 (推荐)

动态性能视图展示了更多更丰富的信息,帮助DBA快速诊断"当前"的阻塞现象. 它还能捕获诸如SQL语句和执行当前SQL语句已经使用的CPU时间, 内存大小,运行总时间,逻辑读数等.

use [master] 
GO 
SELECT 
 DB_NAME(Blocked.database_id) AS 'database', 
 Blocked.Session_ID AS 'blocked SPID', 
 Blocked_SQL.TEXT AS 'blocked SQL', 
 Waits.wait_type AS 'wait resource', 
 Blocking.Session_ID AS 'blocking SPID', 
 Blocking_SQL.TEXT AS 'blocking SQL', 
 sess.status AS 'blocking status', 
 sess.total_elapsed_time AS 'blocking elapsed time', 
 sess.logical_reads AS 'blocking logical reads', 
 sess.memory_usage AS 'blocking memory usage', 
 sess.cpu_time AS 'blocking cpu time', 
 sess.program_name AS 'blocking program', 
 GETDATE() AS 'timestamp' 
FROM sys.dm_exec_connections AS Blocking 
 INNER JOIN sys.dm_exec_requests AS Blocked ON Blocked.Blocking_Session_ID = Blocking.Session_ID 
 INNER JOIN sys.dm_os_waiting_tasks AS Waits ON waits.Session_ID = Blocked.Session_ID 
 INNER JOIN sys.dm_exec_sessions sess ON sess.session_id = Blocking.Session_ID 
 CROSS APPLY sys.dm_exec_sql_text(Blocking.most_recent_sql_handle) AS Blocking_SQL 
 CROSS APPLY sys.dm_exec_sql_text(Blocked.sql_handle) AS Blocked_SQL