Sqlserver
 sql >> Datenbank >  >> RDS >> Sqlserver

So transformieren Sie Daten aus Zeilen basierend auf einer bestimmten Spalte in eine andere Datenstruktur

Sie können das Unpivot/Pivot verwenden, um das gewünschte Ergebnis zu erhalten. Es gibt verschiedene Möglichkeiten, wie Sie das Ergebnis erhalten können. Wenn Sie eine begrenzte Anzahl von Werten haben, können Sie die Abfrage fest codieren, aber wenn Sie eine unbekannte Anzahl von Werten haben, müssen Sie dynamisches SQL verwenden.

Der UNPIVOT-Prozess konvertiert die mehreren Spalten von c1 , etc` in mehrere Zeilen. Sobald sich die Daten in mehreren Zeilen befinden, können Sie die PIVOT-Funktion einfach anwenden. Sie können die Unpivot-Funktion oder CROSS APPLY verwenden, um die Daten aus mehreren Spalten zu konvertieren:

select id,
  col = 'Service'+Service+'_'+col+'_'+cast(seq as varchar(10)),
  value
from
(
  select id, service, c1, cn
    , row_number() over(partition by id
                        order by service) seq
  from yourtable
) t
cross apply
(
  select 'c1', c1 union all
  select 'cn', cn
) c (col, value)

Siehe SQL-Geige mit Demo . Cross Apply konvertiert Ihre Daten in das Format:

| ID |           COL | VALUE |
|  1 | ServiceA_c1_1 |     5 |
|  1 | ServiceA_cn_1 |     3 |
|  1 | ServiceB_c1_2 |     2 |
|  1 | ServiceB_cn_2 |     1 |
|  2 | ServiceA_c1_1 |     9 |
|  2 | ServiceA_cn_1 |     4 |

Sobald die Daten in diesem Format vorliegen, können Sie PIVOT anwenden:

select id, ServiceA_c1_1, ServiceA_cn_1,
  ServiceB_c1_2, ServiceB_cn_2
from
(
  select id,
    col = 'Service'+Service+'_'+col+'_'+cast(seq as varchar(10)),
    value
  from
  (
    select id, service, c1, cn
      , row_number() over(partition by id
                          order by service) seq
    from yourtable
  ) t
  cross apply
  (
    select 'c1', c1 union all
    select 'cn', cn
  ) c (col, value)
) d
pivot
(
  max(value)
  for col in (ServiceA_c1_1, ServiceA_cn_1,
              ServiceB_c1_2, ServiceB_cn_2)
) piv;

Siehe SQL-Geige mit Demo .

Wenn Sie dann eine unbekannte Anzahl von Werten haben, können Sie die obige Abfrage in dynamisches SQL umwandeln:

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

select @cols = STUFF((SELECT ',' + QUOTENAME('Service'+Service+'_'+col+'_'+cast(seq as varchar(10))) 
                    from 
                    (
                      select service, 
                        row_number() over(partition by id
                                          order by service) seq
                      from yourtable 
                    )d
                    cross apply
                    (
                      select 'c1', 1 union all
                      select 'cn', 2
                    ) c (col, so)
                    group by seq, Service, col, so
                    order by seq, so
            FOR XML PATH(''), TYPE
            ).value('.', 'NVARCHAR(MAX)') 
        ,1,1,'')

set @query = 'SELECT id, ' + @cols + ' 
            from 
            (
              select id,
                col = ''Service''+Service+''_''+col+''_''+cast(seq as varchar(10)),
                value
              from
              (
                select id, service, c1, cn
                  , row_number() over(partition by id
                                      order by service) seq
                from yourtable
              ) t
              cross apply
              (
                select ''c1'', c1 union all
                select ''cn'', cn
              ) c (col, value)
            ) x
            pivot 
            (
                max(value)
                for col in (' + @cols + ')
            ) p '

execute sp_executesql @query;

Siehe SQL-Geige mit Demo . Beide geben ein Ergebnis:

| ID | SERVICEA_C1_1 | SERVICEA_CN_1 | SERVICEB_C1_2 | SERVICEB_CN_2 |
|  1 |             5 |             3 |             2 |             1 |
|  2 |             9 |             4 |        (null) |        (null) |