Объедините объекты JSON из нескольких строк в один объект
CREATE TABLE t(Id int, typee nvarchar(50), jsonStr nvarchar(max));
INSERT INTO t(Id, typee, jsonStr) VALUES
(3786, 'APV', '{"1":1,"3":3,"4":24,"5":95}'),
(3786, 'VN', '{"1":3,"5":25}');
-- Expected result
-- {"APV": {"1":1,"3":3,"4":24,"5":95}, "VN":{"1":3,"5":25} }
SELECT Id,(
SELECT CASE WHEN typee = 'VN' THEN jsonStr END AS [VN]
, CASE WHEN typee = 'VO' THEN jsonStr END AS [VO]
, CASE WHEN typee = 'APV' THEN jsonStr END AS [APV]
FROM t AS x
WHERE x.Id = t.Id
FOR JSON AUTO, WITHOUT_ARRAY_WRAPPER
) AS TEST1
FROM t
GROUP BY Id
Хотелось бы получить такой выход:
{
"APV": {
"1": 1,
"3": 3,
"4": 24,
"5": 95
},
"VN": {
"1": 3,
"5": 25
}
}
2 ответа
Решение
Проблема в том, что вы сохраняете JSON, а затем форматируете и возвращаете его как JSON. Вам нужно будет хранить данные, отличные от JSON, чтобы делать это так, как вы хотите. Поэтому его проще рассматривать как строку:
SELECT t.id,
'{' + STRING_AGG( '"' +t.typee + '": ' + t.jsonStr,',') WITHIN GROUP (ORDER BY typee) + '}'
FROM t
GROUP BY t.id;
С помощью FOR XML PATH
:
SELECT t1.id,
'{' + STUFF((SELECT ',"' + t2.typee + '": ' + t2.jsonStr
FROM t t2
WHERE t2.Id = t1.id
FOR XML PATH(N''),TYPE).value('.','nvarchar(MAX)'),1,1,'') + '}'
FROM t t1
GROUP BY t1.id;
FOR JSON
будет рассматривать строку как строку, даже если она представляет действительный JSON. Вам нужно использоватьJSON_QUERY
для преобразования строки JSON в фактический объект JSON. MIN
требуется объединить несколько строк в одну:
SELECT Id, (
SELECT JSON_QUERY(MIN(CASE WHEN typee = 'APV' THEN jsonStr END)) AS [APV]
, JSON_QUERY(MIN(CASE WHEN typee = 'VN' THEN jsonStr END)) AS [VN]
, JSON_QUERY(MIN(CASE WHEN typee = 'VO' THEN jsonStr END)) AS [VO]
FROM t AS x
WHERE x.id = t.id
FOR JSON AUTO, WITHOUT_ARRAY_WRAPPER
)
FROM t
GROUP BY Id