MSSQL XML性能问题

时间:2016-09-23 07:20:45

标签: sql-server xml xpath xquery sqlxml

示例数据:

<Data>
  <row preis="2.4000" anzahl="72.0000" rabatt="0.0000" mwst="16.0000" bez="tägliche Standreinigung" adoku_nr="1694351" adr_nr="95" anmpos_nr="-1" doku_nr="1694351" gen="0" pos="12" preis_vk="2.4000" preis_lp="2.4000" druck="1" artikel_nr="601" artposarten_nr="1" />
  <row preis="330.0000" anzahl="1.0000" rabatt="0.0000" mwst="15.0000" bez="Abwasseranschluß" adoku_nr="500570" adr_nr="95" anmpos_nr="1" doku_nr="500570" gen="0" pos="6" preis_vk="330.0000" preis_lp="330.0000" druck="1" artikel_nr="304" artposarten_nr="1" />
</Data>

我的目标是为adoku_nr(行的id),属性名称,属性值设置列。

我通过

实现了这一目标
select
  T.C.value('(../@adoku_nr)[1]', 'int') as ID,
  T.C.value('local-name(.)', 'nvarchar(128)') as Name,
  T.C.value('(.)[1]', 'nvarchar(max)') as Value
from @Data.nodes('/Data/row/@*') as T(C)

结果:

ID      Name    Value
1694351 preis   2.4000
1694351 anzahl  72.0000
1694351 rabatt  0.0000
1694351 mwst    16.0000
1694351 bez tägliche Standreinigung
1694351 adoku_nr    1694351
1694351 adr_nr  95
1694351 anmpos_nr   -1
1694351 doku_nr 1694351
1694351 gen 0
1694351 pos 12
1694351 preis_vk    2.4000
1694351 preis_lp    2.4000
1694351 druck   1
1694351 artikel_nr  601
1694351 artposarten_nr  1
500570  preis   330.0000
500570  anzahl  1.0000
500570  rabatt  0.0000
500570  mwst    15.0000
500570  bez Abwasseranschluß
500570  adoku_nr    500570
500570  adr_nr  95
500570  anmpos_nr   1
500570  doku_nr 500570
500570  gen 0
500570  pos 6
500570  preis_vk    330.0000
500570  preis_lp    330.0000
500570  druck   1
500570  artikel_nr  304
500570  artposarten_nr  1

现在我的问题是如果我在更大的xml上执行此查询,在我的情况下是120行,它变得超级慢(4秒),问题似乎是ID列,没有它回到更低的毫秒级别。 有没有办法优化此查询以实现相同的目标,但具有更好的性能?

1 个答案:

答案 0 :(得分:2)

您可以将此需求作为2015_09_19_03_30添加到XQuery

查看XPath

[@adoku_nr=sql:variable("@rowId")]

从我的魔法玻璃灯泡更新

行上有declare @data XML= '<Data> <row preis="2.4000" anzahl="72.0000" rabatt="0.0000" mwst="16.0000" bez="tägliche Standreinigung" adoku_nr="1694351" adr_nr="95" anmpos_nr="-1" doku_nr="1694351" gen="0" pos="12" preis_vk="2.4000" preis_lp="2.4000" druck="1" artikel_nr="601" artposarten_nr="1" /> <row preis="999" anzahl="999" rabatt="9999" mwst="16.0000" bez="tägliche Standreinigung" adoku_nr="999" adr_nr="95" anmpos_nr="-1" doku_nr="1694351" gen="0" pos="12" preis_vk="2.4000" preis_lp="2.4000" druck="1" artikel_nr="601" artposarten_nr="1" /> </Data>'; declare @rowId varchar(100)='1694351'; --try with "999" here to get the other row select T.C.value('(../@adoku_nr)[1]', 'int') as ID, T.C.value('local-name(.)', 'nvarchar(128)') as Name, T.C.value('(.)[1]', 'nvarchar(max)') as Value from @Data.nodes('/Data/row[@adoku_nr=sql:variable("@rowId")]/@*') as T(C) ,行的属性为.nodes(),您直接向下走。不需要昂贵的后向导航......

我认为你正在寻找这个:

CROSS APPLY r.nodes()