一直没有认真了解UPDATE操作的锁。近期在MSDN论坛上看到一个问题,询问堆表更新的死锁问题,问题非常easy,有相似这种表及数据:

CREATE TABLE dbo.tb(

c1 int,

c2 char(10),

c3 varchar(10)

);

GO

DECLARE @id int;

SET @id = 0;

WHILE @id <5

BEGIN;

SET @id = @id + 1;

INSERT dbo.tb VALUES( @id, 'b' + RIGHT(10000 + @id, 4), 'c' + RIGHT(100000 + @id, 4) );

END;

在查询一中运行更新操作:

BEGIN TRAN

UPDATE dbo.tb SET c2 = 'xx' WHERE c1 = 2;

WAITFOR DELAY '00:00:30';

UPDATE dbo.tb SET c2 = 'xx' WHERE c1 = 5;

ROLLBACK;

在查询一运行開始后,立即在查询二中运行以下的操作

BEGIN TRAN

UPDATE dbo.tb SET c2 = 'xx' WHERE c1 = 1;

ROLLBACK;

为什么会出现死锁,假设条件改为 c1 = 4 则不会死锁。

開始的时候想得比較简单,死锁的表现是形成循环等待(对于两个查询而言,能够简单地觉得就是在相互等待对方锁定资源的释放)。

对于这个样例而言。第一个查询更新两次,会先更新并锁定一条记录,然后等待第二个更新。但第二个查询仅仅会更新一条记录。它要么与第一个查询冲突,无法获得锁。须要等待查询一完毕,这个时候它并没有锁定什么;要么能够获得锁,完毕更新。

似乎不应该会出现死锁,死锁会不会是其它原因导致。

在自己的电脑上简单測试了一下。似乎也确实没有死锁。

但后面通过Profile跟踪更新操作的下锁情况才发现。自己的分析大错特错了。

主要原因在于没有正确理解更新操作是怎样用锁的。

在联机帮助上“锁模式”中有关于更新的U(更新锁)和X(排它锁)的说明

http://msdn.microsoft.com/zh-cn/library/ms175519(v=sql.105).aspx

只是说得确实挺模糊的。里面还提到了S锁。我一直以为是查询数据过程中用的S锁(也 SELECT 一样)。找到满足条件的记录后用U锁,再转换为X锁做更新。

Profile(事件探查器)跟踪的结果让我知道了这是一个错误的理解,在Profile中新建一个跟踪,选择Locks中的Lock:Acquired (加锁),Lock:Acquired(释放锁)解两个事件,在筛选中设置仅仅跟踪測试用的查询窗体相应的spid(能够运行 PRINT @@SPID 获得),然后运行一个更新语句。比方 UPDATE dbo.tb SET c2 = 'xx' WHERE c1 = 3

在Profile中能够看到。对于每条记录都有加 U 锁的操作,对于不满足条件的记录,会立即释放U锁;对于满足条件的记录,终于转换为X锁。例如以下图所看到的。

注意一下,在这个跟踪结果里面。并没有出现S锁。

另外学做了一些測试:

  1. 通过加大记录量做更新測试,会发现数据扫描涉及的记录都有U锁,并不限于更新记录所在的页。这从还有一个角度说明了大表中Scan 可怕。

  2. 当使用索引Scan的时候,也会通过跟踪发现所Scan的索引资源有U锁,假设更新不涉及索引变化。那以仅仅会相应的记录有U转X锁。索引的U锁会释放;假设影响索引,那么索引的U锁会转X锁。

  3. 删除操作与更新操作相似

  4. 使用 UPDATE aSET c2 = 'xx' FROM dbo.tb AS a WITH(NOLOCK) WHERE c1 = 3  的加锁情况是一样的。 并不会由于NOLOCK的提示而不加 U 或者 X 锁


最后回头研究一下演示样例中的死锁问题:

  • 对于查询一,第一个更新依次扫描表中全部记录,对于每条记录,加 U 锁,推断是否符合更新条件。假设符合,转换为 X 锁;假设不符合条件。释放 U 锁。第一个更新完毕的时候,查询一锁定了一条记录(由于事务未完毕,所以锁一直保持),然后等待第二个更新

  • 对于查询二,依次扫描表中的每条记录(与前面的更新一样),假设它更新的记录在查询一更新的记录前被扫描到,那么这条记录也会变成 X 锁;当继续并进行到查询一的X锁记录的零点,U 与 X 冲突,无法继续,这时候查询二等待查询一释放锁

  • 查询一的第二个更新開始运行。依次扫描每条记录。同一个事务内不会有冲突。所以它不会与自己之前锁定的记录有冲突,但进行到查询二锁定的记录的时候,它也无法获得 U 锁。它须要等待查询二释放资源。

    这个时候就形成了相互等待,符合死锁条件

  • 假设查询二须要更新的记录在查询一的第一个更新记录之后。则不会有死锁。由于查询二在扫描到查询一第一个更新的记录时就会由于锁冲突等待了,这个时候它没有对不论什么记录设置与查询一的操作有冲突的锁。我自己測试的时候没有死锁,就是这种情况。

    注意这里面提到的顺序。是数据读取的顺序,不一定与存储顺序一样,磁盘上记录的顺序也不一定与INSERT的记录顺序一样,这也是我用相同条件没有測试出死锁的原因(我的环境中,恰好读出的顺序与INSERT的顺序不一样)

更新时,记录读取的顺序,能够通过Profile跟踪的Lock:Acquired (加锁)事件来看。涉及大量数据时,假设server支持。还会有并发读取。这也是分析死锁时要考虑的因素



初步了解更新锁(U)与排它锁(X)的更多相关文章

  1. Sql Server 锁 排它锁 更新锁 共享锁

    引用别人的.有时间整体整理下. 引用地址:http://www.cnblogs.com/wenjl520/archive/2012/08/24/2654412.html 锁的概述 一. 为什么要引入锁 ...

  2. 在SQL Server里为什么我们需要更新锁

    今天我想讲解一个特别的问题,在我每次讲解SQL Server里的锁和阻塞(Locking & Blocking)都会碰到的问题:在SQL Server里,为什么我们需要更新锁?在我们讲解具体需 ...

  3. 使用Sqlserver更新锁防止数据脏读

    有时候我们需要控制某条记录在程序读取后就不再进行更新,直到事务执行完释放后才可以.这时候我们就可以将所有要操作当前记录的查询加上更新锁,以防止查询后被其它事务修改.这种操作只锁定表中某行而不会锁定整个 ...

  4. SqlServer中的更新锁(UPDLOCK)

    UPDLOCK.UPDLOCK 的优点是允许您读取数据(不阻塞其它事务)并在以后更新数据,同时确保自从上次读取数据后数据没有被更改.当我们用UPDLOCK来读取记录时可以对取到的记录加上更新锁,从而加 ...

  5. 数据库更新锁WITH UPDLOCK

    今天因为并发的问题,又讨论了一遍.之前以为同时两个线程开启,线程A加了更新锁,线程B没有加,线程A更新后,线程B也会继续下去代码.但是今天测试了一下,原来线程A更新后(解锁),线程B将不会继续,会出现 ...

  6. 【SqlServer】SqlServer中的更新锁(UPDLOCK)

    UPDLOCK.UPDLOCK 的优点是允许您读取数据(不阻塞其它事务)并在以后更新数据,同时确保自从上次读取数据后数据没有被更改.当我们用UPDLOCK来读取记录时可以对取到的记录加上更新锁,从而加 ...

  7. 【转】Mysql学习---MySQL悲观锁中的排它锁

    [原文]https://www.toutiao.com/i6595305814087434760/ 悲观锁中的排它锁. 排它锁关键字:for update 特点:会锁住行或者表,防止其他事务进行修改操 ...

  8. SqlServer中的更新锁(UPDLOCK和READPAST)

    UPDLOCK和READPAST,通过UPDLOCK和READPAST的结合我们能够解决许多问题,比如我当前项目中对于更新预约人数,则用到了UPDLOCK和READPAST,因为考虑到并发如果固定预约 ...

  9. 分布式锁 ----zookeeper实践 (排它锁)

    排它锁概念: Exclusive Locks,被称为X锁,写锁,独占锁.如果事物T1对数据对象O1加上了排它锁,那么在整个加锁期间,只允许事务T1对O1进行读写操作,其他事务必须等到T1释放锁后才能进 ...

随机推荐

  1. FastDfs java客户端上传、删除文件

    #配置文件 connect_timeout = 2 network_timeout = 30 charset = UTF-8 http.tracker_http_port = 9090 http.an ...

  2. java 符号引用与直接引用

    简单来说: 符号引用就是字符串,这个字符串包含足够的信息,以供实际使用时可以找到相应的位置.你比如说某个方法的符号引用,如:“java/io/PrintStream.println:(Ljava/la ...

  3. jdk,tomcat,mvn,android,php,linux等的初始化配置

    jdk配置:系统变量->新建->变量名:JAVA_HOME 变量值:c:\jdk1.6.0_21(jdk安装目录:C:\Program Files (x86)\Java\jdk1.7.0_ ...

  4. Cow Dance Show

    题目大意: 经过几个月的排练,奶牛们基本准备好展出她们的年度舞蹈表演.今年她们要表演的是著名的奶牛芭蕾——“cowpelia”. 表演唯一有待决定的是舞台的尺寸.一个大小为K的舞台可以支持K头牛同时在 ...

  5. 推送通知iOS客户端编写实现及推送服务器端编写

    http://blog.csdn.net/tonny_guan/article/details/8963262 1.iOS客户端编程 推送通知技术在Mac OS X和iOS系统上都可以运行,我们本章主 ...

  6. Couchbase应用示例(初探)

    安装过程:略. 1. 新建Web项目 从NuGet获取并引用: CouchbaseNetClient,添加后引用列表显示为 : Couchbase.NetClient 2. 需要对项目添加引用,这里我 ...

  7. JAVA版拆分大整数为2幂的和算法

    import java.util.ArrayList; import java.util.List; public class StrTest { public static void main(St ...

  8. J.U.C并发框架源码阅读(九)LinkedBlockingQueue

    基于版本jdk1.7.0_80 java.util.concurrent.LinkedBlockingQueue 代码如下 /* * ORACLE PROPRIETARY/CONFIDENTIAL. ...

  9. 51nod 1137.矩阵乘法-矩阵乘法

    1137 矩阵乘法 基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题   给出2个N * N的矩阵M1和M2,输出2个矩阵相乘后的结果.   Input 第1行:1个数N, ...

  10. Python的程序结构[5] -> 模块/Module[0] -> 内建模块 builtins

    builtins 内建模块 / builtins Module 在Python的模块中,有一种特殊模块,无需导入便可以使用,其中包含了许多内建函数与类. builtins 模块内容 / builtin ...