如何使用GROUP BY连接SQL Server中的string?

如何得到:

id Name Value 1 A 4 1 B 8 2 C 9 

 id Column 1 A:4, B:8 2 C:9 

不需要CURSOR,WHILE循环或用户自定义function

只需要对FOR XML和PATH进行创新。

[注意:此解决scheme仅适用于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 

使用XMLpath将不会像您所期望的那样完美地连接在一起……它将用“&amp;”replace“&” 也会惹上<" and "> …也许还有一些其他的事情,不知道…但你可以试试这个

我遇到了这个解决方法…你需要replace:

 FOR XML PATH('') ) 

有:

 FOR XML PATH(''),TYPE ).value('(./text())[1]','VARCHAR(MAX)') 

…或NVARCHAR(MAX)如果这就是你正在使用的。

为什么地狱不SQL有一个连接集合函数? 这是一个PITA。

当我尝试将Kevin Fairchild的build议转换为包含空格和特殊XML字符(&,<,>)的string时,遇到了一些问题。

我的代码的最终版本(不回答原始问题,但可能对某人有用)如下所示:

 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 eg &gt; &lt; etc*/ ).value('.','VARCHAR(MAX)') ,1,2,'') as NameValues FROM #YourTable Results GROUP BY ID DROP TABLE #YourTable 

而不是使用空格作为分隔符,并用逗号replace所有空格,它只是预先为每个值预留一个逗号和空格,然后使用STUFF删除前两个字符。

XML编码通过使用TYPE指令自动处理。

另一个select使用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 

SQL Server 2005和更高版本允许您创build自己的自定义聚合函数 ,包括像串联这样的东西 – 请参阅链接文章底部的示例。

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; 

如果是SQL Server 2017或SQL Server Vnext,可以使用SQL Azure中的string_agg,如下所示:

 select id, string_agg(concat(name, ':', [value]), ', ') from #YourTable group by id 

这种问题在这里经常被问到,解决scheme将很大程度上取决于底层需求:

https://stackoverflow.com/search?q=sql+pivot

https://stackoverflow.com/search?q=sql+concatenate

通常情况下,如果没有dynamicsql,用户定义的函数或游标,就没有SQL的唯一方法。

为了增加凯德所说的,这通常是一个前端显示的东西,因此应该在那里处理。 我知道,有时在SQL中用于文件导出或其他“仅限SQL”解决scheme的东西比写100%更容易,但大多数情况下,这种连接应该在显示层处理。

八年后… Microsoft SQL Server vNext数据库引擎终于增强了Transact-SQL以直接支持分组string连接。 社区技术预览版本1.0添加了STRING_AGG函数,CTP 1.1为STRING_AGG函数添加了WITHIN GROUP子句。

参考: https : //msdn.microsoft.com/en-us/library/mt775028.aspx

在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 

不需要光标… 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 

这只是Kevin Fairchild的post的一个补充(非常聪明)。 我会添加它作为评论,但我没有足够的积分:)

我正在使用这个想法来处理我正在处理的视图,但是我所介绍的项目包含空格。 所以我稍微修改了代码,不要使用空格作为分隔符。

再次感谢非常酷的解决方法凯文!

 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 

让我们变得非常简单:

 SELECT stuff( ( select ', ' + x from (SELECT 'xxx' x union select 'yyyy') tb FOR XML PATH('') ) , 1, 2, '') 

replace这一行:

 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