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

Consulta de tabela cruzada com colunas dinâmicas no SQL Server 2005 para cima


Existem duas maneiras de executar um PIVOT estático onde você codifica os valores e dinâmico onde as colunas são determinadas quando você executa.

Mesmo que você queira uma versão dinâmica, às vezes é mais fácil começar com um PIVOT estático e, em seguida, trabalhar para um dinâmico.

Versão estática:
SELECT studentid, name, sex,[C], [C++], [English], [Database], [Math], total, average
from 
(
  select s1.studentid, name, sex, subjectname, score, total, average
  from Score s1
  inner join
  (
    select studentid, sum(score) total, avg(score) average
    from score
    group by studentid
  ) s2
    on s1.studentid = s2.studentid
) x
pivot 
(
   min(score)
   for subjectname in ([C], [C++], [English], [Database], [Math])
) p

Consulte SQL Fiddle com demonstração

Agora, se você não conhece os valores que serão transformados, pode usar o SQL dinâmico para isso:
DECLARE @cols AS NVARCHAR(MAX),
    @query  AS NVARCHAR(MAX)

select @cols = STUFF((SELECT distinct ',' + QUOTENAME(SubjectName) 
                    from Score
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')



set @query = 'SELECT studentid, name, sex,' + @cols + ', total, average
              from 
             (
                select s1.studentid, name, sex, subjectname, score, total, average
                from Score s1
                inner join
                (
                  select studentid, sum(score) total, avg(score) average
                  from score
                  group by studentid
                ) s2
                  on s1.studentid = s2.studentid
            ) x
            pivot 
            (
                min(score)
                for subjectname in (' + @cols + ')
            ) p '

execute(@query)

Consulte SQL Fiddle with Demo

Ambas as versões produzirão os mesmos resultados.

Apenas para completar a resposta, se você não tiver um PIVOT função, então você pode obter este resultado usando CASE e uma função agregada:
select s1.studentid, name, sex, 
  min(case when subjectname = 'C' then score end) C,
  min(case when subjectname = 'C++' then score end) [C++],
  min(case when subjectname = 'English' then score end) English,
  min(case when subjectname = 'Database' then score end) [Database],
  min(case when subjectname = 'Math' then score end) Math,
  total, average
from Score s1
inner join
(
  select studentid, sum(score) total, avg(score) average
  from score
  group by studentid
) s2
  on s1.studentid = s2.studentid
group by s1.studentid, name, sex, total, average

Consulte SQL Fiddle with Demo