研究死锁,或者观察sp_lock,有时候最恼人的莫过于你看到下面研究成果的key lock,但是却不知道究竟是哪个page 哪个row被lock住了:

Exec sp_lock:


 
 
就说上面的key (9dd27be994c0) 吧,能不能知道这个key,究竟是对应于那个table,那个data page,甚至哪一行(row)呢?

可以的。且听我慢慢说来。

先说这一行:

52        20        978102525      2          KEY      (9dd27be994c0)                      X          GRANT

其中20就是dbid了,indid 2表示是nonclustered  index,而978102525就是objectid了。这个比较好懂。你可以使用Select object_name(978102525)来得到表的名字:

而Key (9dd27be994c0)其实是键值的哈希(hash)值。Hash出来的值,你无法逆向得到它的键值。但是幸运的是,你可以使用微软的undocumented的宏,%%lockres%%,来对表的所有键进行一次同样的hash运算,然后你就可以从hash的结果集里面找到你想要的键值啦。首先,需要知道index为2的是那个index:

select name,index_id,type_desc from sys.indexes where object_id=object_id('test')

name           
index_id    type_desc

--------------- ----------- ---------------------

cidx1           1           CLUSTERED

idx1            2           NONCLUSTERED

然后就可以使用%%lockres%%了:

SELECT   %%lockres%% 'key', test.*  FROM  test with(index(idx1))

或者:

SELECT   %%lockres%% 'key', test.*  FROM  test with(index=2)

结果如下:

你可以看到,对应sp_lock 的key的hash为9dd27be994c0的行,就是c1=2的那行。简单点你甚至可以使用下面 的语句直接得到某个key hash值的行:

SELECT   %%lockres%% 'key', test.*  FROM  test with(index(idx1)) where %%lockres%% ='(9dd27be994c0)'

再进一步,能否得到该行对应的page和fileid呢?这个就需要使用另一个undocumented的宏%%physloc%%了:

select %%physloc%%,test.*  from test with(index(idx1))

上面的返回值是一个十六进制的值,表示page,file,和slot id。可以使用sys.fn_physlocformatter函数将它转换成一个可读的(file:page:slot)格式的值:

select %%physloc%%,sys.fn_physlocformatter(%%physloc%%) 'physical location',test.*  from test with(index(idx1))

结果如下:

对应第二行数据的,就在37828这个page了。这个时候,你可以使用DBCC PAGE来打印出它的内容,探究slot为2 的那行:

dbcc traceon(3604)

dbcc page(20,1,37828,1)

结果:

PAGE: (1:37828)

BUFFER:

BUF @0x00000000803BB400

bpage = 0x000000002CAA6000         
bhash = 0x0000000000000000         
bpageno = (1:37828)

bdbid = 20                         
breferences = 0                    
bcputicks = 0

bsampleCount = 0                   
bUse1 = 42097                       bstat = 0x10b

blog = 0x5ab21c9a                  
bnext = 0x0000000000000000

PAGE HEADER:

Page @0x000000002CAA6000

m_pageId = (1:37828)               
m_headerVersion = 1                
m_type = 2

m_typeFlagBits = 0x0                m_level = 0                         m_flagBits = 0x0

m_objId (AllocUnitId.idObj) = 194  
m_indexId (AllocUnitId.idInd) = 256

Metadata: AllocUnitId = 72057594050641920

Metadata: PartitionId = 72057594046251008                                Metadata: IndexId = 2

Metadata: ObjectId = 978102525     
m_prevPage = (0:0)                 
m_nextPage = (0:0)

pminlen = 5                        
m_slotCnt = 5                      
m_freeCnt = 7997

m_freeData = 217                    m_reservedCnt = 0                   m_lsn = (17704:185:18)

m_xactReserved = 0                 
m_xdesId = (0:20808688)            
m_ghostRecCnt = 0

m_tornBits = -1577081955           
DB Frag ID = 1

Allocation Status

GAM (1:2) = ALLOCATED              
SGAM (1:3) = ALLOCATED

PFS (1:32352) = 0x60 MIXED_EXT ALLOCATED   0_PCT_FULL                    DIFF (1:6) = CHANGED

ML (1:7) = NOT MIN_LOGGED

DATA:

Slot 0, Offset 0x60, Length 16, DumpStyle BYTE

Record Type = INDEX_RECORD         
Record Attributes =  NULL_BITMAP
VARIABLE_COLUMNS

Record Size = 16

Memory Dump @0x000000005166A060

0000000000000000:   36010000
00030000 01001000 61626364          
6...........abcd

Slot 1, Offset 0x70, Length 22, DumpStyle BYTE

Record Type = INDEX_RECORD         
Record Attributes =  NULL_BITMAP
VARIABLE_COLUMNS

Record Size = 22

Memory Dump @0x000000005166A070

0000000000000000:   36010000
00030000 02001200 16006162 63640100 
6.............abcd..

0000000000000014:   0000                                         
..

Slot 2, Offset 0xc9, Length 16, DumpStyle BYTE

Record Type = INDEX_RECORD          Record Attributes =  NULL_BITMAP VARIABLE_COLUMNS

Record Size = 16

Memory Dump @0x000000005166A0C9

0000000000000000:   36020000
00030000 01001000 31323334          
6...........1234

Slot 3, Offset 0x96, Length 18, DumpStyle BYTE

Record Type = INDEX_RECORD         
Record Attributes =  NULL_BITMAP
VARIABLE_COLUMNS

Record Size = 18

Memory Dump @0x000000005166A096

0000000000000000:   36030000
00030000 01001200 65656566 6666     
6...........eeefff

Slot 4, Offset 0xa8, Length 17, DumpStyle BYTE

Record Type = INDEX_RECORD         
Record Attributes =  NULL_BITMAP
VARIABLE_COLUMNS

Record Size = 17

Memory Dump @0x000000005166A0A8

0000000000000000:   36040000
00030000 01001100 38397070 70       
6...........89ppp

OFFSET TABLE:

Row - Offset

4 (0x4) - 168 (0xa8)

3 (0x3) - 150 (0x96)

2 (0x2) - 201 (0xc9)

1 (0x1) - 112 (0x70)

0 (0x0) - 96 (0x60)

DBCC execution completed. If DBCC printed error messages, contact your system administrator.

注意,这个行就是index record,和indid=2完全吻合。

参考:http://blogs.msdn.com/b/apgcdsd/archive/2014/08/20/key-lock.aspx

Key lock 的秘密的更多相关文章

  1. 关于InnoDB的Next-Key lock

    最近一段时间在准备新员工培训的材料,本来打算介绍介绍概念就OK的,但是既然写了事务的章节,就特别想介绍一下锁,介绍了锁,就忍不住想介绍一下Next-Key Lock. 大家知道,标准的事务隔离级别有R ...

  2. How to make remote key fob for 2002 BMW 3 series

    Here share with you on how to make remote key fob for 2002 BMW 3 series: Method 1: 1. Working within ...

  3. C# 多线程(lock,Monitor,Mutex,同步事件和等待句柄)

    本篇从 Monitor,Mutex,ManualResetEvent,AutoResetEvent,WaitHandler 的类关系图开始,希望通过本篇的介绍能对常见的线程同步方法有一个整体的认识,而 ...

  4. [MySQL] gap lock/next-key lock浅析

    当InnoDB在判断行锁是否冲突的时候, 除了最基本的IS/IX/S/X锁的冲突判断意外, InnoDB还将锁细分为如下几种子类型: record lock (RK) 记录锁, 仅仅锁住索引记录的一行 ...

  5. gap lock/next-key lock浅析 Basic-Paxos协议日志同步应用

    http://www.cnblogs.com/renolei/p/4673842.html 当InnoDB在判断行锁是否冲突的时候, 除了最基本的IS/IX/S/X锁的冲突判断意外, InnoDB还将 ...

  6. gap lock/next-key lock浅析Basic-Paxos协议日志同步应用

    http://www.cnblogs.com/renolei/p/4673842.html 当InnoDB在判断行锁是否冲突的时候, 除了最基本的IS/IX/S/X锁的冲突判断意外, InnoDB还将 ...

  7. 1.Redis Lock

    使用场景 同步锁,让业务方法在锁设定的时间内是同步执行的 redisService.setIfAbsent redisService.expire @PostMapping("/update ...

  8. WPF 依赖属性与依赖对象

    在介绍依赖属性之前,我先介绍下属性的历史 属性的历史:      早期C++的类中,只有字段及方法,暴露数据靠的是方法, 但是字段直接暴露会不安全,所以才用方法来暴露,在设置的时候加些约束,在MFC中 ...

  9. 开源框架是如何使用设计模式的-MyBatis缓存机制之装饰者模式

    写在前面 聊一聊MyBatis是如何使用装饰者模式的,顺便回顾下缓存的相关知识,可以看看右侧目录一览内容概述. 装饰者模式 这里就不了它的概念了,总结下就是套娃.利用组合的方式将装饰器组合进来,增强共 ...

随机推荐

  1. 腾讯RTX登录提示失败问题及处理办法

    出现问题时图片: 解决步骤: 首先ping一下RTX管理器所在的的ip地址 telnet一下服务器的端口 8000,8010 是否能通. 具体操作: 在无法登陆的客户端电脑上开始-运行-输入cmd 确 ...

  2. 东软实训2-在jsp中使用javaBean

    在JSP中可以像使用普通类一样访问JavaBean,在脚本元素中实例化类的对象,调用对象的方法.JSP提供了3个动作元素,和来访问JavaBean. 1.1 动作用于初始化JavaBean,或者定位一 ...

  3. php提取身份证号码中的生日日期以及验证是否为未成年人的函数

    php 提取身份证号码中的生日日期以及确定是否成年的一个函数.可以同时确定15位和18位的身份证,经本人亲测,非常好用,分享函数代码如下: <?php //用php从身份证中提取生日,包括15位 ...

  4. Flask学习记录之MarkDown编辑文本

    为了让网页支持markdown编辑文本,使用如下了4个库 PageDown : 在前端提供一个可以实时将markdown内容转换成html文本进行效果预览的编辑器 Flask-PageDown: 这个 ...

  5. CSS太阳月亮地球三角恋旋转效果

    纯粹玩一下,好像没有什么实际的卵用,but,纯玩买不了上当,纯玩买不了受骗........ 地月旋转的一个css效果,无聊玩玩,可以复制到记事本试试 <!DOCTYPE html>< ...

  6. Python学习笔记整理(五)Python中的列表.

    列表和字段,这两种类型几乎是Python所有脚本的主要工作组件.他们都可以在原处进行修改,可以按需求增加或缩短,而且包含任何种类的对象或者被嵌套. 一.列表 列表的主要属性: *任意对象的有序集合 从 ...

  7. Java学习笔记--String StringBuffer StringBuilder

    String StringBuffer StringBuilder String http://docs.oracle.com/javase/7/docs/api/ 中文: http://www.cn ...

  8. Sublime Text 2 注册码

    ----- BEGIN LICENSE ----- Andrew Weber Single User License EA7E- 813A03DD 5E4AD9E6 6C0EEB94 BC99798F ...

  9. 从汇编来看c语言

    一. 学习过程 从C语言的角度提出一些问题,这些问题再从汇编的角度考虑,还真的很有意思. (1) 我们用高级语言编程时,一般不可能不用到变量,但是一定要用到变量吗?还有这些变量从汇编的角度是怎么实现的 ...

  10. Bash 使用技巧大补贴

    https://linuxtoy.org/archives/the-best-tips-and-tricks-for-bash.html