二、查看锁的活动
下面演示一个实例,它使用sys.dm_tran_locks动态视图监视数据库中锁的活动。
打开一个查询窗口,执行如下语句:
代码如下:
USE AdventureWorks
BEGIN TRAN
SELECT ProductID, ModifiedDate
FROM Production.ProductDocument
WITH (TABLOCKX)
打开另一个查询窗口,执行:
代码如下:
SELECT request_session_id sessionid,
resource_type type,
resource_database_id dbid,
OBJECT_NAME(resource_associated_entity_id, resource_database_id) objectname,
request_mode rmode,
request_status rstatus
FROM sys.dm_tran_locks
WHERE resource_type IN ('DATABASE', 'OBJECT')
执行结果:
代码如下:
/*
sessionid type dbid objectname rmode rstatus
51 DATABASE 4 NULL S GRANT
52 DATABASE 4 NULL S GRANT
53 DATABASE 8 NULL S GRANT
56 DATABASE 8 NULL S GRANT
53 OBJECT 8 ProductDocument X GRANT
*/
解析:本示例中,我们首先启动了一个新事务,并使用TABLOCKX锁提示(这个提示对表放置了排他锁),对Production.ProductDocument表执行了一个查询。查询sys.dm_tran_locks动态管理视力可以监视当前SQL Server实例中打开了哪些锁。它返回了AdventureWorks数据库中活动锁的列表。可以在结果中的最后一行看到ProductDocument表上的排他锁。
前三列定义了会话锁、资源类型和数据库ID。第四列使用了Object_Name函数,注意它使用了两个参数(对象ID和数据库ID)来指定访问哪个名称(第二个参数是SQL Server 2005 SP2引入的,它用来指定为了转换对象名称而使用哪个数据库)。同时也查询锁定请求模式和状态,最后,From子句引用DMV,用Where子句指定了两个资源类型。Resource_Type指定了锁定的资源类型,如DatabaseObjectFilePageKeyRIDExtentMetadataApplicationAllocation_Unit或HOBT类型。依赖资源类型的resource_associated_entity_id,确定ID是object ID, allocation unit ID, 或Hobt ID。
■如果resource_associated_entity_id列包含Object ID(资源类型为Object),可以使用sys.objects目录视图来转换名称。
■如果resource_associated_entity_id列包含allocation unit ID(资源类型为Allocation_Unit),可以引用sys.allocatiion_units和contain_id联结到sys.partitions上,就可以确定object ID。
■如果resource_associated_entity_id列包含Hobt ID(资源类型为KeypageRow或HOBT),可以直接引用sys.partitions,然后查找相应的Object ID。
(本文来源于图老师网站,更多请访问http://m.tulaoshi.com/bianchengyuyan/)■对于Database、Extent、 Application或MetaData的资源类型,resource_associated_entity_id列将为0。
使用sys.dm_tran_locks能对无法预料的并发问题进行故障调试。例如,一个查询会话占用锁的时间可能比预期时间长而被锁,或者锁的粒度或锁模式不是我们所期望的(可能是希望使用表锁而不是更小粒度的行锁或页锁)。理解锁处于的锁定级别有助于我们更有效地对查询的并发问题进行故障调试。
三、控制表的锁升级行为
每个在SQL Server中创建的锁都会消耗内存资源。当锁的数量增加时,内存就会减少。如果锁的内存使用百分比超过一个特定阈值,SQL Server会将细粒度锁(页或行)转换为粗粒度锁(表锁)。这个过程称为锁升级。锁升级可以减少SQL Server实例占有的锁数量,减少锁内存的使用。
(本文来源于图老师网站,更多请访问http://m.tulaoshi.com/bianchengyuyan/)虽然细粒度会消耗更多的内存,但由于多个查询可以访问未锁定的行,因此也会改善并发性。引入表锁可能会减少内存的消耗,但也会带来阻塞,这是因为一条查询锁住了整个表。根据使用数据库的应用程序,这个行为可能是不希望发生的,而且你可能希望当SQL Server实施锁升级时尽量获得更多的控制。
SQL Server 2008引入了使用ALter table命令在表级别控制锁升级的功能。现在可以从如下3个设置中选择:
■Table 这是SQL Server 2005中使用的默认行为。当设置为该值时,在表级别启用了锁升级,不论是否为分区表。
■Auto 如果表已分区,则在分区级别(堆或B树)启用锁升级。如果表未分区,锁升级将发生在表级别上。
■Disable 在表级别删除锁升级。注意,对于用了TABLOCK 提示或使用可序列化隔离级别下Heap的查询时,你仍然可能看到表锁。
下面示例演示了修改表的新设置:
代码如下:
ALTER TABLE Person.Address
SET (LOCK_ESCALATION = AUTO)
--注意这句在SQL Server 2005下会出错
SELECT lock_escalation,lock_escalation_desc
FROM sys.tables
WHERE name='Address'
/*
lock_escalation lock_escalation_desc
2 AUTO
*/
下来,我们禁用锁升级:
代码如下:
ALTER TABLE Person.Address
SET ( LOCK_ESCALATION = DISABLE)
SELECT lock_escalation,lock_escalation_desc
FROM sys.tables
WHERE name='Address'
/*
lock_escalation lock_escalation_desc
1 DISABLE
*/
说明:在更改了这个配置后,可以通过查询sys.tables目录视图的lock_escalation_desc列来验证这个选项。
注意:如果表未分区,通常情况为表级别升级。如果你指定了Disable选项,将不会出现表级别的锁升级。这会提高并发性,但如果你请求访问大量的行或页,会增加内存的消耗。
邀月 来自 http://www.cnblogs.com/downmoon