SQL XML Problemas explícitos com a criação de árvore xml (pai-filho)
-
21-12-2019 - |
Pergunta
Estou enfrentando alguns problemas com o xml explícito no sql server, onde ele não gera xml de acordo com o relacionamento que especifiquei na consulta sql.A consulta é feita no banco de dados pubs e embora o caminho xml seja mais fácil de usar, meu treinador precisa que ele seja feito em xml explícito.
SELECT 1 AS Tag,
NULL AS Parent,
NULL AS [TitleTypes!1],
NULL AS [TitleType!2!Type],
NULL AS [TitleType!2!AveragePrice],
NULL AS [Title!3!title_id],
NULL AS [Title!3!price]
UNION ALL
SELECT 2,
1,
NULL ,
type AS [TitleType!2!Type],
AVG(price) AS [TitleType!2!AveragePrice],
NULL AS [Title!3!title_id],
NULL AS [Title!3!price]
from titles
GROUP BY type
UNION ALL
SELECT 3,
2,
NULL ,
type AS [TitleType!2!Type],
NULL AS [TitleType!2!AveragePrice],
title_id AS [Title!3!title_id],
price AS [Title!3!price]
from titles
FOR XML EXPLICIT;
A saída que produz:
<TitleTypes>
<TitleType Type="business " AveragePrice="13.7300" />
<TitleType Type="mod_cook " AveragePrice="11.4900" />
<TitleType Type="popular_comp" AveragePrice="21.4750" />
<TitleType Type="psychology " AveragePrice="13.5040" />
<TitleType Type="trad_cook " AveragePrice="15.9633" />
<TitleType Type="UNDECIDED ">
<Title title_id="BU1032" price="19.9900" />
<Title title_id="BU1111" price="11.9500" />
<Title title_id="BU2075" price="2.9900" />
<Title title_id="BU7832" price="19.9900" />
<Title title_id="MC2222" price="19.9900" />
<Title title_id="MC3021" price="2.9900" />
<Title title_id="MC3026" />
<Title title_id="PC1035" price="22.9500" />
<Title title_id="PC8888" price="20.0000" />
<Title title_id="PC9999" />
<Title title_id="PS1372" price="21.5900" />
<Title title_id="PS2091" price="10.9500" />
<Title title_id="PS2106" price="7.0000" />
<Title title_id="PS3333" price="19.9900" />
<Title title_id="PS7777" price="7.9900" />
<Title title_id="TC3218" price="20.9500" />
<Title title_id="TC4203" price="11.9500" />
<Title title_id="TC7777" price="14.9900" />
</TitleType>
</TitleTypes>
A saída que eu quero:
<TitleTypes>
<TitleType Type="business" AveragePrice="11.22">
<Title title_id="BU1111" Price="11.34"/>
<Title title_id="TC7777" Price="14.2"/>
</TitleType>
<TitleType Type="popular_comp" AveragePrice="13.99">
<Title title_id="BU1111" Price="15.9"/>
<Title title_id="TC7777" Price="16.22"/>
</TitleType>
</TitleTypes>
Solução
Normalmente, você não precisa do modo explícito.Você pode gerar quase todos os XML que quiser para caminho xml:
select
t1.type as [@Type],
avg(t1.price) as [@AveragePrice],
(
select
t2.title_id as [@title_id],
t2.price as [@price]
from titles as t2
where t2.type = t1.type
for xml path('Title'), type
)
from titles as t1
group by t1.type
for xml path('TitleType'), root('TitleTypes')
Mas como seu xml é centrado em atributos, é ainda mais fácil de usar para xml bruto:
select
t1.type as [Type],
avg(t1.price) as AveragePrice,
(
select
t2.title_id
t2.price
from titles as t2
where t2.type = t1.type
for xml raw('Title'), type
)
from titles as t1
group by t1.type
for xml raw('TitleType')
Outras dicas
Aqui está a solução que funcionou para mim.Veja a cláusula ORDER BY.Se você estiver usando o SQL Server Management Studio na cláusula ORDER BY, ele mostrará o possível nome da coluna pela qual você pode formatar seu xml de saída conforme desejar.
SELECT 1 AS Tag,
NULL AS Parent,
NULL AS [TitleTypes!1],
NULL AS [TitleType!2!Type],
NULL AS [TitleType!2!AveragePrice],
NULL AS [Title!3!title_id],
NULL AS [Title!3!price]
UNION ALL
SELECT 2,
1,
NULL ,
type AS [TitleType!2!Type],
AVG(price) AS [TitleType!2!AveragePrice],
NULL AS [Title!3!title_id],
NULL AS [Title!3!price]
from titles
GROUP BY type
UNION ALL
SELECT 3,
2,
NULL ,
type AS [TitleType!2!Type],
NULL AS [TitleType!2!AveragePrice],
title_id AS [Title!3!title_id],
price AS [Title!3!price]
from titles
ORDER BY [TitleTypes!1], [TitleType!2!Type], [Title!3!title_id] ,Tag
FOR XML EXPLICIT;
P.S.FOR XML EXPLICIT é uma merda.Mas se você estiver desenvolvendo/atualizando um sistema realmente antigo, você precisa conhecê-lo antes de substituí-lo por técnicas mais novas.