![](/img/trans.png)
[英]How to use GROUP BY to concatenate different columns in SQL Server?
[英]How to use GROUP BY to concatenate strings in SQL Server?
如何得到:
id Name Value
1 A 4
1 B 8
2 C 9
到
id Column
1 A:4, B:8
2 C:9
不需要 CURSOR、WHILE 循環或用戶定義的函數。
只需要對 FOR XML 和 PATH 發揮創意。
[注意:此解決方案僅適用於 SQL 2005 及更高版本。 原始問題沒有指定使用的版本。]
CREATE TABLE #YourTable ([ID] INT, [Name] CHAR(1), [Value] INT)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'A',4)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'B',8)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (2,'C',9)
SELECT
[ID],
STUFF((
SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX))
FROM #YourTable
WHERE (ID = Results.ID)
FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
,1,2,'') AS NameValues
FROM #YourTable Results
GROUP BY ID
DROP TABLE #YourTable
如果是 SQL Server 2017 或 SQL Server Vnext, SQL Azure 你可以使用string_agg
如下:
select id, string_agg(concat(name, ':', [value]), ', ')
from #YourTable
group by id
使用 XML 路徑不會像您預期的那樣完美連接……它將用“&”替換“&” 並且還會弄亂<" and ">
......也許還有其他一些事情,不確定......但你可以試試這個
我遇到了一個解決方法......你需要更換:
FOR XML PATH('')
)
和:
FOR XML PATH(''),TYPE
).value('(./text())[1]','VARCHAR(MAX)')
...或NVARCHAR(MAX)
如果那是您使用的。
為什么SQL
沒有連接聚合函數? 這是一個皮塔餅。
當我嘗試將 Kevin Fairchild 的建議轉換為使用包含空格和特殊 XML 字符( &
、 <
、 >
)的字符串時,我遇到了一些問題。
我的代碼的最終版本(它沒有回答原始問題,但可能對某人有用)如下所示:
CREATE TABLE #YourTable ([ID] INT, [Name] VARCHAR(MAX), [Value] INT)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'Oranges & Lemons',4)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'1 < 2',8)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (2,'C',9)
SELECT [ID],
STUFF((
SELECT ', ' + CAST([Name] AS VARCHAR(MAX))
FROM #YourTable WHERE (ID = Results.ID)
FOR XML PATH(''),TYPE
/* Use .value to uncomment XML entities e.g. > < etc*/
).value('.','VARCHAR(MAX)')
,1,2,'') as NameValues
FROM #YourTable Results
GROUP BY ID
DROP TABLE #YourTable
它不是使用空格作為分隔符並用逗號替換所有空格,而是在每個值前添加一個逗號和空格,然后使用STUFF
刪除前兩個字符。
XML 編碼是通過使用TYPE指令自動處理的。
使用 Sql Server 2005 及更高版本的另一種選擇
---- test data
declare @t table (OUTPUTID int, SCHME varchar(10), DESCR varchar(10))
insert @t select 1125439 ,'CKT','Approved'
insert @t select 1125439 ,'RENO','Approved'
insert @t select 1134691 ,'CKT','Approved'
insert @t select 1134691 ,'RENO','Approved'
insert @t select 1134691 ,'pn','Approved'
---- actual query
;with cte(outputid,combined,rn)
as
(
select outputid, SCHME + ' ('+DESCR+')', rn=ROW_NUMBER() over (PARTITION by outputid order by schme, descr)
from @t
)
,cte2(outputid,finalstatus,rn)
as
(
select OUTPUTID, convert(varchar(max),combined), 1 from cte where rn=1
union all
select cte2.outputid, convert(varchar(max),cte2.finalstatus+', '+cte.combined), cte2.rn+1
from cte2
inner join cte on cte.OUTPUTID = cte2.outputid and cte.rn=cte2.rn+1
)
select outputid, MAX(finalstatus) from cte2 group by outputid
從http://groupconcat.codeplex.com安裝 SQLCLR 聚合
然后你可以寫這樣的代碼來得到你要求的結果:
CREATE TABLE foo
(
id INT,
name CHAR(1),
Value CHAR(1)
);
INSERT INTO dbo.foo
(id, name, Value)
VALUES (1, 'A', '4'),
(1, 'B', '8'),
(2, 'C', '9');
SELECT id,
dbo.GROUP_CONCAT(name + ':' + Value) AS [Column]
FROM dbo.foo
GROUP BY id;
八年后... Microsoft SQL Server vNext 數據庫引擎終於增強了 Transact-SQL 以直接支持分組字符串連接。 Community Technical Preview 1.0 版添加了 STRING_AGG 函數,CTP 1.1 為 STRING_AGG 函數添加了 WITHIN GROUP 子句。
參考: https : //msdn.microsoft.com/en-us/library/mt775028.aspx
SQL Server 2005 及更高版本允許您創建自己的自定義聚合函數,包括諸如連接之類的事情 - 請參閱鏈接文章底部的示例。
在 Oracle 中,您可以使用 LISTAGG 聚合函數。
原始記錄
name type
------------
name1 type1
name2 type2
name2 type3
數據庫
SELECT name, LISTAGG(type, '; ') WITHIN GROUP(ORDER BY name)
FROM table
GROUP BY name
導致
name type
------------
name1 type1
name2 type2; type3
這只是對 Kevin Fairchild 帖子的補充(順便說一下,這很聰明)。 我會把它添加為評論,但我還沒有足夠的分數:)
我將這個想法用於我正在處理的視圖,但是我連接的項目包含空間。 所以我稍微修改了代碼,不使用空格作為分隔符。
再次感謝凱文的酷解決方法!
CREATE TABLE #YourTable ( [ID] INT, [Name] CHAR(1), [Value] INT )
INSERT INTO #YourTable ([ID], [Name], [Value]) VALUES (1, 'A', 4)
INSERT INTO #YourTable ([ID], [Name], [Value]) VALUES (1, 'B', 8)
INSERT INTO #YourTable ([ID], [Name], [Value]) VALUES (2, 'C', 9)
SELECT [ID],
REPLACE(REPLACE(REPLACE(
(SELECT [Name] + ':' + CAST([Value] AS VARCHAR(MAX)) as A
FROM #YourTable
WHERE ( ID = Results.ID )
FOR XML PATH (''))
, '</A><A>', ', ')
,'<A>','')
,'</A>','') AS NameValues
FROM #YourTable Results
GROUP BY ID
DROP TABLE #YourTable
這種問題在這里經常被問到,解決方案將在很大程度上取決於底層需求:
https://stackoverflow.com/search?q=sql+pivot
和
https://stackoverflow.com/search?q=sql+concatenate
通常,如果沒有動態 sql、用戶定義的函數或游標,就沒有僅使用 SQL 的方法來執行此操作。
補充一下 Cade 所說的,這通常是前端顯示的事情,因此應該在那里處理。 我知道有時在 SQL 中編寫 100% 的東西更容易,例如文件導出或其他“僅限 SQL”的解決方案,但大多數情況下,這種連接應該在您的顯示層中處理。
不需要游標......一個while循環就足夠了。
------------------------------
-- Setup
------------------------------
DECLARE @Source TABLE
(
id int,
Name varchar(30),
Value int
)
DECLARE @Target TABLE
(
id int,
Result varchar(max)
)
INSERT INTO @Source(id, Name, Value) SELECT 1, 'A', 4
INSERT INTO @Source(id, Name, Value) SELECT 1, 'B', 8
INSERT INTO @Source(id, Name, Value) SELECT 2, 'C', 9
------------------------------
-- Technique
------------------------------
INSERT INTO @Target (id)
SELECT id
FROM @Source
GROUP BY id
DECLARE @id int, @Result varchar(max)
SET @id = (SELECT MIN(id) FROM @Target)
WHILE @id is not null
BEGIN
SET @Result = null
SELECT @Result =
CASE
WHEN @Result is null
THEN ''
ELSE @Result + ', '
END + s.Name + ':' + convert(varchar(30),s.Value)
FROM @Source s
WHERE id = @id
UPDATE @Target
SET Result = @Result
WHERE id = @id
SET @id = (SELECT MIN(id) FROM @Target WHERE @id < id)
END
SELECT *
FROM @Target
讓我們變得非常簡單:
SELECT stuff(
(
select ', ' + x from (SELECT 'xxx' x union select 'yyyy') tb
FOR XML PATH('')
)
, 1, 2, '')
替換這一行:
select ', ' + x from (SELECT 'xxx' x union select 'yyyy') tb
隨着您的查詢。
如果 group by 主要包含一項,您可以通過以下方式顯着提高性能:
SELECT
[ID],
CASE WHEN MAX( [Name]) = MIN( [Name]) THEN
MAX( [Name]) NameValues
ELSE
STUFF((
SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX))
FROM #YourTable
WHERE (ID = Results.ID)
FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
,1,2,'') AS NameValues
END
FROM #YourTable Results
GROUP BY ID
沒有看到任何交叉應用答案,也不需要 xml 提取。 這是 Kevin Fairchild 所寫的略有不同的版本。 在更復雜的查詢中使用更快、更容易:
select T.ID
,MAX(X.cl) NameValues
from #YourTable T
CROSS APPLY
(select STUFF((
SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX))
FROM #YourTable
WHERE (ID = T.ID)
FOR XML PATH(''))
,1,2,'') [cl]) X
GROUP BY T.ID
使用 Stuff 和 for xml 路徑運算符將行連接到字符串:按兩列分組 -->
CREATE TABLE #YourTable ([ID] INT, [Name] CHAR(1), [Value] INT)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'A',4)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'B',8)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (1,'B',5)
INSERT INTO #YourTable ([ID],[Name],[Value]) VALUES (2,'C',9)
-- retrieve each unique id and name columns and concatonate the values into one column
SELECT
[ID],
STUFF((
SELECT ', ' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX)) -- CONCATONATES EACH APPLICATION : VALUE SET
FROM #YourTable
WHERE (ID = Results.ID and Name = results.[name] )
FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
,1,2,'') AS NameValues
FROM #YourTable Results
GROUP BY ID
SELECT
[ID],[Name] , --these are acting as the group by clause
STUFF((
SELECT ', '+ CAST([Value] AS VARCHAR(MAX)) -- CONCATONATES THE VALUES FOR EACH ID NAME COMBINATION
FROM #YourTable
WHERE (ID = Results.ID and Name = results.[name] )
FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
,1,2,'') AS NameValues
FROM #YourTable Results
GROUP BY ID, name
DROP TABLE #YourTable
使用替換函數和 FOR JSON PATH
SELECT T3.DEPT, REPLACE(REPLACE(T3.ENAME,'{"ENAME":"',''),'"}','') AS ENAME_LIST
FROM (
SELECT DEPT, (SELECT ENAME AS [ENAME]
FROM EMPLOYEE T2
WHERE T2.DEPT=T1.DEPT
FOR JSON PATH,WITHOUT_ARRAY_WRAPPER) ENAME
FROM EMPLOYEE T1
GROUP BY DEPT) T3
有關示例數據和更多方法, 請單擊此處
如果您啟用了 clr,則可以使用 GitHub 中的Group_Concat庫
另一個沒有垃圾的例子: ",TYPE).value('(./text())[1]','VARCHAR(MAX)')"
WITH t AS (
SELECT 1 n, 1 g, 1 v
UNION ALL
SELECT 2 n, 1 g, 2 v
UNION ALL
SELECT 3 n, 2 g, 3 v
)
SELECT g
, STUFF (
(
SELECT ', ' + CAST(v AS VARCHAR(MAX))
FROM t sub_t
WHERE sub_t.g = main_t.g
FOR XML PATH('')
)
, 1, 2, ''
) cg
FROM t main_t
GROUP BY g
輸入輸出是
************************* -> *********************
* n * g * v * * g * cg *
* - * - * - * * - * - *
* 1 * 1 * 1 * * 1 * 1, 2 *
* 2 * 1 * 2 * * 2 * 3 *
* 3 * 2 * 3 * *********************
*************************
我使用了這種可能更容易掌握的方法。 獲取一個根元素,然后連接以選擇具有相同 ID 但不是“官方”名稱的任何項目
Declare @IdxList as Table(id int, choices varchar(max),AisName varchar(255))
Insert into @IdxLIst(id,choices,AisName)
Select IdxId,''''+Max(Title)+'''',Max(Title) From [dbo].[dta_Alias]
where IdxId is not null group by IdxId
Update @IdxLIst
set choices=choices +','''+Title+''''
From @IdxLIst JOIN [dta_Alias] ON id=IdxId And Title <> AisName
where IdxId is not null
Select * from @IdxList where choices like '%,%'
對於我所有的醫療保健人員:
SELECT
s.NOTE_ID
,STUFF ((
SELECT
[note_text] + ' '
FROM
HNO_NOTE_TEXT s1
WHERE
(s1.NOTE_ID = s.NOTE_ID)
ORDER BY [line] ASC
FOR XML PATH(''),TYPE).value('(./text())[1]','VARCHAR(MAX)')
,
1,
2,
'') AS NOTE_TEXT_CONCATINATED
FROM
HNO_NOTE_TEXT s
GROUP BY NOTE_ID
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.