SQL2008中SQL应用之-锁定(locking) 应用分析

一、锁的基本概念:

锁定(Locking)是一个关系型数据库系统的常规和必要的一部分,它防止对相同数据作 并发更新 或在更新过程中查看数据, 从而保证被更新数据的完整性。它也能防止用户读取正在被修改的数据 。Sql Server动态地管理锁定,然而,还是很有必要 了解Transact- SQL查询如何影响SQL Server中的锁定。在此,简单介绍下锁的基本常识。

锁定有助于防止并发问题的发生。当一个用户试图读取另一个用户正在修改的数据,或者修改另一个用户正在读取的数据时,或者尝试修改另一个事务正在尝试修改的数据时,就会出现并发问题。

SQL Server资源会被锁定,资源的锁定方式称作它的锁定模式(lock mode),下表列出SQL Server处理的主要锁定模式:


























名称

描述

共享 (S) 用于不更改或不更新数据的读取操作,如 SELECT 语句。
更新 (U) 用于可更新的资源中。 防止当多个会话在读取、锁定以及随后可能进行的资源更新时发生常见形式的死锁。
排他 (X) 用于数据修改操作,例如 INSERT、UPDATE 或 DELETE。 确保不会同时对同一资源进行多重更新。
意向 用于建立锁的层次结构。 意向锁包含三种类型:意向共享 (IS)、意向排他 (IX) 和意向排他共享 (SIX)。
架构 在执行依赖于表架构的操作时使用。 架构锁包含两种类型:架构修改 (Sch-M) 和架构稳定性 (Sch-S)。
大容量更新 (BU) 在向表进行大容量数据复制且指定了 TABLOCK 提示时使用。
键范围 当使用可序列化事务隔离级别时保护查询读取的行的范围。 确保再次运行查询时其他事务无法插入符合可序列化事务的查询的行。

可以锁定SQL Server中的各种对象,既可以是一个行,也可以是一个表或数据库。可以锁定的资源在粒度(granularity)上差异很大。从细(行)到粗(数据库)。细粒度锁允许更大的数据库并发,因为用户能对某些未锁定的行执行查询。然而,每个由SQL Server产生的锁都需要内存,所以数以千计独立的行级别的锁也会影响SQL Server的性能。粗粒度的锁降低了并发性,但消耗的资源也较少。下表介绍SQL Server可以锁定的资源:






































资源

说明

KEY 索引中用于保护可序列化事务中的键范围的行锁。
PAGE 数据库中的 8 KB 页,例如数据页或索引页。
EXTENT 一组连续的八页,例如数据页或索引页。
HoBT 堆或 B 树。 用于保护没有聚集索引的表中的 B 树(索引)或堆数据页的锁。
TABLE 包括所有数据和索引的整个表。
FILE 数据库文件。
RID 用于锁定堆中的单个行的行标识符。
APPLICATION 应用程序专用的资源。
METADATA 元数据锁。
ALLOCATION_UNIT 分配单元。
DATABASE 整个数据库。

不是所有的锁都能彼此兼容。例如,一个被排他锁锁定的资源不能被再加其他锁。其他事务必须等待或超时,直到排他锁被释放。被更新锁锁定的资源只能接受其他事务的共享锁。被共享锁锁定的资源还能接受其他的共享锁或更新锁。

SQL Server自动分配和升级锁。升级意味着细粒度的锁(行或页锁)被转化为粗粒度的表锁。当单个T-SQL语句在单个表或索引上获取5000多个锁,或者SQL Server实例中的锁数量超过可用内存阈值时,SQL Server会尝试启动锁升级。锁占用系统内存,因此把很多锁转化为一个较大的锁能释放内存资源。然而,在释放内存资源的同时会降低并发性。

SQL Server 2008带来了新的表选项,可以禁用锁升级或在分区(而不是表)范围启用锁升级。


二、查看锁的活动

下面演示一个实例,它使用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指定了锁定的资源类型,如Database\Object\File\Page\Key\RID\Extent\Metadata\Application\Allocation_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(资源类型为Key\page\Row或HOBT),可以直接引用sys.partitions,然后查找相应的Object ID。

■对于Database、Extent、 Application或MetaData的资源类型,resource_associated_entity_id列将为0。

使用sys.dm_tran_locks能对无法预料的并发问题进行故障调试。例如,一个查询会话占用锁的时间可能比预期时间长而被锁,或者锁的粒度或锁模式不是我们所期望的(可能是希望使用表锁而不是更小粒度的行锁或页锁)。理解锁处于的锁定级别有助于我们更有效地对查询的并发问题进行故障调试。

三、控制表的锁升级行为

每个在SQL Server中创建的锁都会消耗内存资源。当锁的数量增加时,内存就会减少。如果锁的内存使用百分比超过一个特定阈值,SQL Server会将细粒度锁(页或行)转换为粗粒度锁(表锁)。这个过程称为锁升级。锁升级可以减少SQL Server实例占有的锁数量,减少锁内存的使用。

虽然细粒度会消耗更多的内存,但由于多个查询可以访问未锁定的行,因此也会改善并发性。引入表锁可能会减少内存的消耗,但也会带来阻塞,这是因为一条查询锁住了整个表。根据使用数据库的应用程序,这个行为可能是不希望发生的,而且你可能希望当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

(0)

相关推荐

  • SQL2008中SQL应用之-锁定(locking) 应用分析

    一.锁的基本概念: 锁定(Locking)是一个关系型数据库系统的常规和必要的一部分,它防止对相同数据作 并发更新 或在更新过程中查看数据, 从而保证被更新数据的完整性.它也能防止用户读取正在被修改的数据 .Sql Server动态地管理锁定,然而,还是很有必要 了解Transact- SQL查询如何影响SQL Server中的锁定.在此,简单介绍下锁的基本常识. 锁定有助于防止并发问题的发生.当一个用户试图读取另一个用户正在修改的数据,或者修改另一个用户正在读取的数据时,或者尝试修改另一个事务

  • SQL2008中SQL应用之- 死锁(Deadlocking)

    在另一方释放资源前,会话1和会话2都不可能继续.所以,SQL Server会选择死锁中的一个会话作为"死锁牺牲品". 注意:死锁牺牲品的会话会被杀死,事务会被回滚. 注意:死锁与正常的阻塞是两个经常被混淆的概念. 发生死锁的一些原因: 1.应用程序以不同的次序访问表.例如会话1先更新了客户然后更新了订单,而会话2先更新了订单然后更新了客户.这就增加了死锁的可能性. 2.应用程序使用了长时间的事务,在一个事务中更新很多行或很多表.这样增加了行的"表面积",从而导致死锁

  • SQL2008中SQL应用之-阻塞(Blocking)应用分析

    通常短时间的阻塞没有问题,且是较忙的应用程序所需要的.然而,设计糟糕的应用程序会导致长时间的阻塞,这就不必要地锁定了资源,而且阻塞了其他会话读取和更新它们. 在SQL Server中,一个阻塞的进程会无限期地保持阻塞,或者直到它超时(根据set lock_timeout).服务器关闭.进程被杀死.连接完成了更新或者其他发生在原始事务上的操作导致它释放了资源上的锁. 发生长时间阻塞的原因如下: 1.在一个没有索引的表上的过量的行锁会导致SQL Server得到一个锁,从而阻塞其他事务. 2.应用程

  • Oracle在Mybatis中SQL语句的配置方法

    数据库中有下划线的字段在实体中应采用驼峰命名法,如P_NAME对应pName,实例如下: 1.XML文件中SQL语句配置(Geteway.xml文件) <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-m

  • oracle数据库中sql%notfound的用法详解

    SQL%NOTFOUND 是一个布尔值.与最近的sql语句(update,insert,delete,select)发生交互,当最近的一条sql语句没有涉及任何行的时候,则返回true.否则返回false.这样的语句在实际应用中,是非常有用的.例如要update一行数据时,如果没有找到,就可以作相应操作.如: begin update table_name set salary = 10000 where emp_id = 10; if sql%notfound then insert into

  • Oracle数据库中SQL语句的优化技巧

    在SQL语句优化过程中,我们经常会用到hint,现总结一下在SQL优化过程中常见Oracle HINT的用法: 1. /*+ALL_ROWS*/ 表明对语句块选择基于开销的优化方法,并获得最佳吞吐量,使资源消耗最小化. 例如: SELECT /*+ALL+_ROWS*/ EMP_NO,EMP_NAM,DAT_IN FROM BSEMPMS WHERE EMP_NO='SCOTT'; 2. /*+FIRST_ROWS*/ 表明对语句块选择基于开销的优化方法,并获得最佳响应时间,使资源消耗最小化.

  • Oracle中SQL语句连接字符串的符号使用介绍

    Oracle中SQL语句连接字符串的符号为|| 复制代码 代码如下: select catstr(tcdm) || (',') from T_YWCJ_RWCJR where cjrjh='009846' and rwid='12050' and jsdm='CJY' 拼接成一条数据并连接一个","

  • 细数MySQL中SQL语句的分类

    1:数据定义语言(DDL)   用于创建.修改.和删除数据库内的数据结构,如:1:创建和删除数据库(CREATE DATABASE || DROP  DATABASE):2:创建.修改.重命名.删除表(CREATE  TABLE || ALTER TABLE|| RENAME TABLE||DROP  TABLE):3:创建和删除索引(CREATEINDEX  || DROP INDEX)   2:数据查询语言(DQL)   从数据库中的一个或多个表中查询数据(SELECT)   3:数据操作语

  • Mybatis中 SQL语句复用

    mapper.xml 中共用 mapper.xml 间共用 项目中也许我们会遇到一段sql语句被多个查询.增加等语句用到的情况,如何去偷懒呢,复用sql无疑是较好的选择 这里只提供简单的示范: 如果只是单表查询,并且希望共用的sql只会出现在同一个mapper.xml文件中,那么我们可以直接在 <mapper namespace="XXXXX"></mapper> 中写下面的业务代码 <sql id="unitSql"> a.us

  • C#中SQL参数传入空值报错解决方案

    C#中的null与SQL中的NULL是不一样的,SQL中的NULL用C#表示出来就是DBNull.Value. 注意:SQL参数是不能接受C#的null值的,传入null就会报错. 下面我们看个例子: SqlCommand cmd=new SqlCommand("Insert into Student values(@StuName,@StuAge)" ,conn); cmd.parameters.add("@StuName" ,stuName); cmd.para

随机推荐