Sql server中內(nèi)部函數(shù)fn_PhysLocFormatter存在解析錯誤詳解
前言
有網(wǎng)友指出,SQL Server 2012中fn_PhysLocFormatter內(nèi)部函數(shù)在解析數(shù)據(jù)行記錄位置時存在錯誤,見:http://www.itpub.net/thread-1751655-1-1.html,實(shí)際測試后發(fā)現(xiàn),一是2008R2中同樣存在問題,二是不僅頁號解析存在問題,槽號解析也存在同樣問題。
下面先查看表NT_SiteInfo的數(shù)據(jù)行記錄位置。
select SiteID,%%physloc%%,sys.fn_PhysLocFormatter(%%physloc%%) from NT_SiteInfo
SiteID |
%%physloc%% |
sys.fn_PhysLocFormatter(%%physloc%%) |
1 |
0xE900000001000000 |
(1:59648:0) |
23 |
0xE900000001000100 |
(1:59648:1) |
24 |
0xE900000001000200 |
(1:59648:2) |
|
...... |
|
149 |
0xE900000001007F00 |
(1:59648:127) |
150 |
0xE900000001008000 |
(1:59648:128) |
151 |
0xE900000001008100 |
(1:59648:33024) |
152 |
0xE900000001008200 |
(1:59648:33280) |
|
...... |
|
226 |
0xE90000000100CC00 |
(1:59648:52224) |
227 |
0xE90000000100CD00 |
(1:59648:52480) |
228 |
0x4B02000001000000 |
(1:587:0) |
229 |
0x4B02000001000100 |
(1:587:1) |
|
...... |
|
360 |
0x4B02000001007F00 |
(1:587:127) |
361 |
0x4B02000001008000 |
(1:587:128) |
362 |
0x4B02000001008100 |
(1:587:33024) |
363 |
0x4B02000001008200 |
(1:587:33280) |
|
...... |
|
422 |
0x4B0200000100BD00 |
(1:587:48384) |
423 |
0x4B0200000100BE00 |
(1:587:48640) |
424 |
0x3C05000001000000 |
(1:1340:0) |
425 |
0x3C05000001000100 |
(1:1340:1) |
|
...... |
|
552 |
0x3C05000001008000 |
(1:1340:128) |
553 |
0x3C05000001008100 |
(1:1340:33024) |
|
|
|
596 |
0x3C0500000100AC00 |
(1:1340:44032) |
597 |
0x9978000001000000 |
(1:39288:0) |
|
...... |
|
658 |
0x9978000001003D00 |
(1:39288:61) |
下面查看表NT_SiteInfo分配的數(shù)據(jù)頁情況。
dbcc ind(wjgk,nt_siteinfo,0)
PagePID |
IAMFID |
IAMPID |
PageType |
IndexLevel |
NextPagePID |
PrevPagePID |
238 |
NULL |
NULL |
10 |
NULL |
0 |
0 |
233 |
1 |
238 |
1 |
0 |
587 |
0 |
587 |
1 |
238 |
1 |
0 |
1340 |
233 |
1340 |
1 |
238 |
1 |
0 |
30873 |
587 |
30873 |
1 |
238 |
1 |
0 |
0 |
1340 |
Microsoft未公開的偽列%%physloc%%,類型為Binary(8),返回表中記錄的RowID,格式是:前4字節(jié)表示頁號,中間2字節(jié)表示文件號,最后2字節(jié)表示槽號。
對照上面的實(shí)際數(shù)據(jù),可以發(fā)現(xiàn)sys.fn_PhysLocFormatter
在解析記錄位置時,既有采用高字節(jié)在前的BIG_ENDIAN格式,又有采用低字節(jié)在前的LITTLE_ENDIAN格式,造成采用高字節(jié)在前的BIG_ENDIAN格式解析的數(shù)據(jù)錯誤:
頁號解析:
E9000000解析為59648(E900),錯誤,實(shí)際應(yīng)為233(E9)
4B020000解析為576(24B),正確
3C050000解析為1340(53C),正確
99780000解析為39288(9978),錯誤,實(shí)際應(yīng)為30873(7899)
槽號解析:
8000解析為128(0080),正確
8100解析為33024(8100),錯誤,應(yīng)為129(0081)
下面給出錯誤原因。
先看下sys.fn_PhysLocFormatter
函數(shù)的定義:
select OBJECT_DEFINITION(object_id('sys.fn_PhysLocFormatter')) go ------------------------------------------------------------------------------- -- Name: sys.fn_PhysLocFormatter -- -- Description: -- Formats the output of %%physloc%% virtual column -- -- Notes: ------------------------------------------------------------------------------- create function sys.fn_PhysLocFormatter (@physical_locator binary (8)) returns varchar (128) as begin declare @page_id binary (4) declare @file_id binary (2) declare @slot_id binary (2) -- Page ID is the first four bytes, then 2 bytes of page ID, then 2 bytes of slot -- select @page_id = convert (binary (4), reverse (substring (@physical_locator, 1, 4))) select @file_id = convert (binary (2), reverse (substring (@physical_locator, 5, 2))) select @slot_id = convert (binary (2), reverse (substring (@physical_locator, 7, 2))) return '(' + cast (cast (@file_id as int) as varchar) + ':' + cast (cast (@page_id as int) as varchar) + ':' + cast (cast (@slot_id as int) as varchar) + ')' end
再看下reverse函數(shù):
select reverse('工人') ---- 人工 (1 行受影響) select reverse('12345工人') --------- 人工54321 (1 行受影響) select reverse('工12345人') --------- 人54321工 (1 行受影響)
結(jié)論:問題出在reverse函數(shù)上。
reverse函數(shù)的作用是字符反轉(zhuǎn),而不是字節(jié)反轉(zhuǎn),當(dāng)遇到81-FE之間的字節(jié)時,被認(rèn)為是雙字節(jié)字符而組合在一起參與反轉(zhuǎn)操作,造成了錯誤。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
SQL Server阻止保存修改表結(jié)構(gòu)的解決方法
修改DeUser表中列的數(shù)據(jù)類型從varchar修改為int時,進(jìn)行保存時彈出的提示框,阻止保存修改表結(jié)構(gòu),怎么辦?這篇文章就為大家詳細(xì)介紹了解決SQL Server阻止保存修改表結(jié)構(gòu)問題的方法,感興趣的小伙伴們可以參考一下2016-05-05sqlserver Union和SQL Union All使用方法
UNION 操作符用于合并兩個或多個 SELECT 語句的結(jié)果集。2010-09-09SQL Server設(shè)置主鍵自增長列(使用sql語句實(shí)現(xiàn))
主鍵自增長列在進(jìn)行數(shù)據(jù)插入的時候,很有用的,如可以獲取返回的自增ID值,接下來將介紹SQL Server如何設(shè)置主鍵自增長列,感興趣的朋友可以了解下,希望本文對你有所幫助2013-01-01SQL Server 數(shù)據(jù)庫管理常用的SQL和T-SQL語句
SQL Server 數(shù)據(jù)庫管理常用的SQL和T-SQL語句...2007-03-03與 SQL Server 建立連接時出現(xiàn)與網(wǎng)絡(luò)相關(guān)的或特定于實(shí)例的錯誤
與 SQL Server 建立連接時出現(xiàn)與網(wǎng)絡(luò)相關(guān)的或特定于實(shí)例的錯誤,未找到或無法訪問服務(wù)器,請驗(yàn)證實(shí)例名稱是否正確并且 SQL Server 已配置為允許遠(yuǎn)程連接2024-04-04實(shí)例學(xué)習(xí)SQL的Select命令
實(shí)例學(xué)習(xí)SQL的Select命令...2006-12-12