sql >> Databáze >  >> RDS >> Sqlserver

SQL Server 2012 PIVOT bez agregace

Existuje několik způsobů, jak můžete transformovat data. Některé používají agregační funkci a jiné ne. Ale i když otáčíte řetězec, stále můžete použít agregaci.

Agregovat s CASE:

select name,
  max(case when category = 'A' then 'X' else '' end) CategoryA,
  max(case when category = 'B' then 'X' else '' end) CategoryB,
  max(case when category = 'C' then 'X' else '' end) CategoryC,
  max(case when category = 'D' then 'X' else '' end) CategoryD
from yourtable 
group by name

Viz SQL Fiddle s ukázkou

Statické pivotování:

Stále můžete používat PIVOT funkce pro transformaci dat, i když hodnoty jsou řetězce. Pokud máte známý počet kategorií, můžete dotaz napevno zadat:

select name, 
  coalesce(A, '') CategoryA, 
  coalesce(B, '') CategoryB, 
  coalesce(C, '') CategoryC, 
  coalesce(C, '') CategoryD
from
(
  select name, category, 'X' flag
  from yourtable
) d
pivot
(
  max(flag)
  for category in (A, B, C, D)
) piv

Viz SQL Fiddle s ukázkou .

Dynamické pivotování:

Pokud máte neznámý počet kategorií, můžete použít dynamické SQL:

DECLARE @cols AS NVARCHAR(MAX),
    @colsNull AS NVARCHAR(MAX),
    @query  AS NVARCHAR(MAX)

select @cols = STUFF((SELECT  ',' + QUOTENAME(category) 
                    from yourtable
                    group by category
                    order by category
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')

select @colsNull = STUFF((SELECT  ', coalesce(' + QUOTENAME(category)+', '''') as '+QUOTENAME('Category'+category)
                    from yourtable
                    group by category
                    order by category
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')


set @query = 'SELECT name, ' + @colsNull + ' 
              from 
             (
                select name, category, ''X'' flag
                from yourtable
            ) x
            pivot 
            (
                max(flag)
                for category in (' + @cols + ')
            ) p '

execute(@query)

Viz SQL Fiddle s ukázkou .

Vícenásobná spojení:

select c1.name,
  case when c1.category is not null then 'X' else '' end as CategoryA,
  case when c2.category is not null then 'X' else '' end as CategoryB,
  case when c3.category is not null then 'X' else '' end as CategoryC,
  case when c4.category is not null then 'X' else '' end as CategoryD
from yourtable c1
left join yourtable c2
  on c1.name = c2.name
  and c2.category = 'B'
left join yourtable c3
  on c1.name = c3.name
  and c3.category = 'C'
left join yourtable c4
  on c1.name = c4.name
  and c4.category = 'D'
where c1.category = 'A'

Viz SQL Fiddle s ukázkou

Všechny dotazy poskytnou výsledek:

| NAME | CATEGORYA | CATEGORYB | CATEGORYC | CATEGORYD |
--------------------------------------------------------
|  Joe |         X |         X |           |         X |
| Mary |         X |           |         X |         X |



  1. Oracle :Grant Vytvořit tabulku v jiném schématu?

  2. Jak zapsat výstup příkazu SQL do souboru CSV?

  3. Nelze spustit MySql Utilities

  4. Funkce PostgreSQL pro iteraci/působení na mnoha řádcích se stavem