Sqlserver
 sql >> Base de Dados >  >> RDS >> Sqlserver

Achatamento de uma tabela de 1 linha em uma tabela de pares de valores-chave


Uma versão onde não há dinâmica envolvida. Se você tiver nomes de colunas inválidos para usar como nomes de elementos em XML, isso falhará.
select T2.N.value('local-name(.)', 'nvarchar(128)') as [Key],
       T2.N.value('text()[1]', 'nvarchar(max)') as Value
from (select *
      from TableA
      for xml path(''), type) as T1(X)
  cross apply T1.X.nodes('/*') as T2(N)

Uma amostra de trabalho:
declare @T table
(
  Column1 varchar(10), 
  Column2 varchar(10), 
  Column3 varchar(10)
)

insert into @T values('V1','V2','V3')

select T2.N.value('local-name(.)', 'nvarchar(128)') as [Key],
       T2.N.value('text()[1]', 'nvarchar(max)') as Value
from (select *
      from @T
      for xml path(''), type) as T1(X)
  cross apply T1.X.nodes('/*') as T2(N)

Resultado:
Key                  Value
-------------------- -----
Column1              V1
Column2              V2
Column3              V3

Atualizar

Para uma consulta com mais de uma tabela, você pode usar for xml auto para obter os nomes das tabelas no XML. Observe que, se você usar alias para nomes de tabelas na consulta, obterá o alias.
select X2.N.value('local-name(..)', 'nvarchar(128)') as TableName,
       X2.N.value('local-name(.)', 'nvarchar(128)') as [Key],
       X2.N.value('text()[1]', 'nvarchar(max)') as Value
from (
     -- Your query starts here
     select T1.T1ID,
            T1.T1Col,
            T2.T2ID,
            T2.T2Col
     from T1
       inner join T2
         on T1.T1ID = T2.T1ID
     -- Your query ends here
     for xml auto, elements, type     
     ) as X1(X)
  cross apply X1.X.nodes('//*[text()]') as X2(N)

SQL Fiddle