SQL Double Pivot с суммой - PullRequest
       7

SQL Double Pivot с суммой

0 голосов
/ 27 апреля 2018

У меня есть набор данных в SQL Server Management Studio 17, который требует некоторого поворота, чтобы свести несколько строк в одну строку для удобства чтения и экономии места. Для каждого уникального идентификатора существует набор кодов, которые действуют как операторы для значений, с целью отображения результатов в итоговом столбце в конце. Я выполнил основной круг с одним набором кодов, но, похоже, мне нужно сделать второй, чтобы выполнить дальнейшее свертывание / объединение.

Вот некоторые неучтенные данные

PO_NUM   u_id   CODE_1   CODE_2   VALUE
---------------------------------------
0M274316 1      3        9;13     150
0N274316 1      9        9        200000
0O274316 1      6        9        210000
0P274316 1      6        13       21000
0Q274316 1      6        9        50000.5
0R274316 1      15       9        0
0M274317 2      3        9;13     150
0N274317 2      9        9        300000
0O274317 2      6        9        220000
0P274317 2      6        13       22000
0Q274317 2      6        8        90000.5
0R274317 2      15       8        0

С примером поворота с первым уникальным идентификатором в качестве примера я получаю:

PO_NUM    uid  CODE_1   CODE_2    6         9        15
-----------------------------------------------------
0M274316  1      3       9;13    210000    200000   0
0M274316  1              9       21000     NULL     NULL
0M274316  1              13      50000.5   NULL     NULL

Что мне нужно сделать, так это еще один стержень, чтобы в качестве столбцов использовались '9; 13' с суммами, используемыми со значениями. У меня есть эта начальная команда поворота как:

SELECT PO_NUM, CODE_1, CODE_2, VALUE from table
Pivot (max(VALUE) for CODE_2 in ([6],[9],[15])) pvt1

Что объединяет данные, как ожидалось, когда я не включаю значения CODE_1. Но попытка добавить второй свод с CODE_2 только еще больше запутывает данные. Как мне повернуть во второй раз, используя сводные данные из первой команды? Ожидаемый результат будет:

PO_NUM    uid    6         9        15    9;13    9(2)    13   total
------------------------------------------------------------------------
0M274316  1   281000.5   200000     0      150    null   null   481150.5

По сути, они были сведены в одну строку с кодами, разбитыми на столбцы. В действительности столбцы будут переименованы, чтобы избежать конфликтов с именами, поэтому я использовал 9 (2), чтобы показать, что данные все еще там.

Извините за правки.

Ответы [ 2 ]

0 голосов
/ 03 мая 2018

Вы можете попробовать это.

DECLARE @MyTable TABLE(PO_NUM VARCHAR(10),  u_id int,  CODE_1  varchar(10), CODE_2 varchar(10),  VALUE DECIMAL(18,2))
INSERT INTO @MyTable VALUES
('0M274316', 1, '3', '9;13', 150),
('0N274316', 1, '9', '9', 200000),
('0O274316', 1, '6', '9', 210000),
('0P274316', 1, '6', '13', 21000),
('0Q274316', 1, '6', '9', 50000.5),
('0R274316', 1, '15', '9', 0),
('0M274317', 2, '3', '9;13', 150),
('0N274317', 2, '9', '9', 300000),
('0O274317', 2, '6', '9', 220000),
('0P274317', 2, '6', '13', 22000),
('0Q274317', 2, '6', '8', 90000.5),
('0R274317', 2, '15', '8', 0)


SELECT *, [6] + [9] + [15] + [9;13] + [9(2)] + [13] as total FROM (

    SELECT PO_NUM,u_id, CODE + ( CASE WHEN DENSE_RANK() OVER( PARTITION BY CODE ORDER BY C ) > 1 THEN '(' + CAST (  C  AS varchar(50)) + ')' ELSE '' END ) CODE, VALUE 
    FROM (
        SELECT MIN(PO_NUM) OVER(PARTITION BY u_id) PO_NUM, u_id, 1 C, CODE_1 CODE, VALUE FROM @MyTable 
        UNION ALL
        SELECT MIN(PO_NUM) OVER(PARTITION BY u_id) PO_NUM, u_id, 2 C, CODE_2 CODE, VALUE from @MyTable 
    ) T
) SRC
PIVOT(SUM(VALUE) FOR CODE IN ( [6], [9], [15], [9;13], [9(2)], [13])) PVT

Результат:

PO_NUM     u_id        6            9           15        9;13      9(2)        13          total
---------- ----------- ------------ ----------- --------- --------- ----------- ----------- -------------
0M274316   1           281000.50    200000.00   0.00      150.00    460000.50   21000.00    962151.00
0M274317   2           332000.50    300000.00   0.00      150.00    520000.00   22000.00    1174150.50
0 голосов
/ 27 апреля 2018

Если я правильно понимаю, просто используйте условное агрегирование:

select po_num, u_id, code_2,
       max(case when code_1 = 3 then value end) as [3],
       max(case when code_1 = 6 then value end) as [6],
       max(case when code_1 = 9 then value end) as [9],
       max(case when code_1 = 15 then value end) as [15]
from t
group by po_num, u_id, code_2;

Я думаю, это будет выражаться с помощью pivot как:

SELECT PO_NUM, u_id, CODE_2, [3], [6], [9], [15]
from table
Pivot (max(VALUE) for CODE_2 in ([3], [6], [9], [15])) pvt1;
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...