初步了解更新锁(U)与排它锁(X)

??

一直没有认真了解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支持。还会有并发读取。这也是分析死锁时要考虑的因素

??

时间: 2024-08-25 05:04:10

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

lock(3)——更新锁(U)与排它锁(X)

以下文章中详细介绍了update操作过程中更新锁及排它锁的分配情况 http://blog.csdn.net/zjcxc/article/details/27351779 按照以上文章中的追踪方式,发现其实文章lock(2)——创建及更新表过程中SQL SERVER锁资源分配情况中我们通过sys.dm_tran_locks动态视图查询出来的结果都是最终加锁的情况,并没有体现其中锁变化的情况.如果要想详细的知道锁的变化情况,还是使用profile的方式. 我这里为了加深印象与理解,就再将邹大师的测

sql server中的锁 事务锁 更新锁 保持锁 共享锁 你知道吗?

锁定数据库的一个表 SELECT * FROM table WITH (HOLDLOCK) 注意: 锁定数据库的一个表的区别 SELECT * FROM table WITH (HOLDLOCK) 其他事务可以读取表,但不能更新删除 SELECT * FROM table WITH (TABLOCKX) 其他事务不能读取表,更新和删除 SELECT 语句中"加锁选项"的功能说明 SQL Server提供了强大而完备的锁机制来帮助实现数据库系统的并发性和高性能.用户既能使用SQL Ser

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

今天我想讲解一个特别的问题,在我每次讲解SQL Server里的锁和阻塞(Locking & Blocking)都会碰到的问题:在SQL Server里,为什么我们需要更新锁?在我们讲解具体需要的原因前,首先我想给你介绍下当更新锁(Update(U)Lock)获得时,根据它的兼容性锁本身是如何应对的. 一般来说,当执行UPDATE语句时,SQL Server会用到更新锁(Update Lock).如果你查看对应的执行计划,你会看到它包含3个部分 读取数据 计算新值 写入数据 在查询计划的第1部分

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

有时候我们需要控制某条记录在程序读取后就不再进行更新,直到事务执行完释放后才可以.这时候我们就可以将所有要操作当前记录的查询加上更新锁,以防止查询后被其它事务修改.这种操作只锁定表中某行而不会锁定整个表,体验更好. 测试sql代码如下: 在一个查询中执行如下语句 begin tran SELECT InvestState FROM InvestOrdersABC WITH (UPDLOCK) where id=10005 waitfor delay '00:00:10' update Inves

共享锁(S锁)和排它锁(X锁)

共享锁(S锁)和排它锁(X锁) 分类: 数据库2008-11-07 10:53 4409人阅读 评论(2) 收藏 举报 共享锁[S锁]又称读锁,若事务T对数据对象A加上S锁,则事务T可以读A但不能修改A,其他事务只能再对A加S锁,而不能加X锁,直到T释放A上的S锁.这保证了其他事务可以读A,但在T释放A上的S锁之前不能对A做任何修改. 排他锁[X锁]又称写锁.若事务T对数据对象A加上X锁,事务T可以读A也可以修改A,其他事务不能再对A加任何锁,直到T释放A上的锁.这保证了其他事务在T释放A上的锁

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

排它锁概念: Exclusive Locks,被称为X锁,写锁,独占锁.如果事物T1对数据对象O1加上了排它锁,那么在整个加锁期间,只允许事务T1对O1进行读写操作,其他事务必须等到T1释放锁后才能进行操作.在单机环境中,JDK提供了synchronized关键字和ReentrantLock 重用锁来提供排它锁的功能. zookeeper实现排它锁原理: 在需要获取排它锁时,所有的客户端都会调用create方法在固定路径下创建节点,并发环境下,只有一个客户端可以创建成功,相当于获取了锁,当该客户

SqlServer中的更新锁(UPDLOCK)

优点: 允许读取数据(不阻塞其它事务)并在以后更新数据,同时确保自从上次读取数据后数据没有被更改 当用UPDLOCK来读取记录时可以对取到的记录加上更新锁,从而加上锁的记录在其它的线程中是不能更改的只能等本线程的事务结束后才能更改 begin transelect * from address WITH (UPDLOCK) where [Name]='Z'waitfor delay '00:00:10' update address set [Name]='ZZ'commit tran 注意:

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

[原文]https://www.toutiao.com/i6595305814087434760/ 悲观锁中的排它锁. 排它锁关键字:for update 特点:会锁住行或者表,防止其他事务进行修改操作或者获取锁操作. 详见:步步揭秘:MySQL共享锁的特性-2 解决办法 该解决办法就是通过排它锁:for update进行操作 上述SQL可以通过下图进行展现: 线程1:在A时刻开始事务,并将goods_id为1的那一行锁住(行锁),然后进行查询,获取库存. 线程2:在B时刻也想进行同样的操作,先

数据库锁解析

目录 1. 锁的概念 1. 锁的定义 2. 锁的解释 2. 锁的分类 1. 按锁的功能来划分 2. 按锁的概念来划分 3. 锁的关键字 4. 死锁 1. 死锁实战 2. 死锁原理 3. 死锁解决方案 5. 无锁查询 1. 举例实战 2. 原理解析 7. 何时加锁 8. 锁的粒度 1. 定义 2. 页锁 3. 行锁 4. 表锁 9. 锁与事物隔离级别的优先级 10. 锁类型比较 1. HOLDLOCK与TABLOCK 2. TABLOCKX与XLOCK 11. 锁的超时等待 12. 各种锁的兼容关