Создать хранимую процедуру для агрегирования строк - PullRequest
0 голосов
/ 25 августа 2018

Наличие таблицы транзакций со следующими строками:

Id    UserId   PlatformId    TransactionTypeId
-------------------------------------------------
 0       1          3                1
 1       1          1                2
 2       2          3                2
 3       3          2                1
 4       2          3                1

Как мне написать хранимую процедуру, которая может объединять строки в новую таблицу следующего формата?

Id    UserId   Platforms          TransactionTypeId
-------------------------------------------------
 0      1      {"p3":1,"p1":1}    {"t1":1,"t2":1}
 1      2      {"p3":2}           {"t2":1,"t1":1}
 3      3      {"p2":1}           {"t1":1}

Таким образом, строки сгруппированы пользователем, подсчитывают каждый тип платформы / транзакции и сохраняют как строку ключ / значение json.

Ссылка: Мой предыдущий связанный вопрос

Ответы [ 2 ]

0 голосов
/ 25 августа 2018

Ваш образец JSON на самом деле не является JSON, но, поскольку вы хотите его таким образом:

SELECT u.UserId, plt.pValue, ttyp.ttValue
FROM Users AS [u]
CROSS APPLY (
SELECT '{'+STUFF( (SELECT ',"'+pn.pName+'":'+LTRIM(STR(pn.pCount))
    FROM (SELECT p.Name AS pName, COUNT(*) AS pCount 
        FROM transactions t
        left JOIN Platforms p ON p.PlatformID = t.PlatformId
        WHERE t.UserId = u.UserId
        GROUP BY p.PlatformId, p.Name
        ) pn
    FOR XML PATH('')),1,1,'')+'}'
    ) plt(pValue)
CROSS APPLY (
SELECT '{'+STUFF( (SELECT ',"'+tty.ttName+'":'+LTRIM(STR(tty.ttCount))
    FROM (SELECT tt.Name AS ttName, COUNT(*) AS ttCount 
        FROM transactions t
        left JOIN dbo.TransactionType tt ON tt.TransactionTypeId = t.TransactionTypeID
        WHERE t.UserId = u.UserId
        GROUP BY tt.TransactionTypeId, tt.Name
        ) tty
    FOR XML PATH('')),1,1,'')+'}'
    ) ttyp(ttValue)
WHERE EXISTS (SELECT * FROM transactions t WHERE u.UserId = t.UserId)
ORDER BY UserId;

Образец DBFiddle

0 голосов
/ 25 августа 2018

Вы можете использовать GROUP BY и FOR JSON:

SELECT MIN(ID) AS ID, UserId, MIN(sub.x) AS Platforms, MIN(sub2.x) AS Transactions
FROM tab t
OUTER APPLY (SELECT CONCAT('p', platformId) AS platform, cnt 
            FROM (SELECT PlatformId, COUNT(*) AS cnt
                  FROM tab t2 WHERE t2.UserId = t.UserId  
                  GROUP BY PlatformId) s
              FOR JSON AUTO) sub(x)
OUTER APPLY (SELECT CONCAT('t', TransactiontypeId) AS Transactions, cnt 
            FROM (SELECT TransactiontypeId, COUNT(*) AS cnt
                  FROM tab t2 WHERE t2.UserId = t.UserId  
                  GROUP BY TransactiontypeId) s
              FOR JSON AUTO) sub2(x)
GROUP BY UserId;

DBFiddle Demo

Результат немного другой (массивзначения ключа), но, пожалуйста, рассматривайте его как отправную точку.

...