简体   繁体   English

如何使用 GROUP BY 在 SQL Server 中连接字符串?

[英]How to use GROUP BY to concatenate strings in SQL Server?

How do I get:如何得到:

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

to

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

No CURSOR, WHILE loop, or User-Defined Function needed .不需要 CURSOR、WHILE 循环或用户定义的函数

Just need to be creative with FOR XML and PATH.只需要对 FOR XML 和 PATH 发挥创意。

[Note: This solution only works on SQL 2005 and later. [注意:此解决方案仅适用于 SQL 2005 及更高版本。 Original question didn't specify the version in use.]原始问题没有指定使用的版本。]

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

If it is SQL Server 2017 or SQL Server Vnext, SQL Azure you can use string_agg as below:如果是 SQL Server 2017 或 SQL Server Vnext, SQL Azure 你可以使用string_agg如下:

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

using XML path will not perfectly concatenate as you might expect... it will replace "&" with "&amp;"使用 XML 路径不会像您预期的那样完美连接……它将用“&”替换“&” and will also mess with <" and "> ...maybe a few other things, not sure...but you can try this并且还会弄乱<" and "> ......也许还有其他一些事情,不确定......但你可以试试这个

I came across a workaround for this... you need to replace:我遇到了一个解决方法......你需要更换:

FOR XML PATH('')
)

with:和:

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

...or NVARCHAR(MAX) if thats what youre using. ...或NVARCHAR(MAX)如果那是您使用的。

why the hell doesn't SQL have a concatenate aggregate function?为什么SQL没有连接聚合函数? this is a PITA.这是一个皮塔饼。

I ran into a couple of problems when I tried converting Kevin Fairchild's suggestion to work with strings containing spaces and special XML characters ( & , < , > ) which were encoded.当我尝试将 Kevin Fairchild 的建议转换为使用包含空格和特殊 XML 字符( &<> )的字符串时,我遇到了一些问题。

The final version of my code (which doesn't answer the original question but may be useful to someone) looks like this:我的代码的最终版本(它没有回答原始问题,但可能对某人有用)如下所示:

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

DROP TABLE #YourTable

Rather than using a space as a delimiter and replacing all the spaces with commas, it just pre-pends a comma and space to each value then uses STUFF to remove the first two characters.它不是使用空格作为分隔符并用逗号替换所有空格,而是在每个值前添加一个逗号和空格,然后使用STUFF删除前两个字符。

The XML encoding is taken care of automatically by using the TYPE directive. XML 编码是通过使用TYPE指令自动处理的。

Another option using Sql Server 2005 and above使用 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

Install the SQLCLR Aggregates from http://groupconcat.codeplex.comhttp://groupconcat.codeplex.com安装 SQLCLR 聚合

Then you can write code like this to get the result you asked for:然后你可以写这样的代码来得到你要求的结果:

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;

Eight years later... Microsoft SQL Server vNext Database Engine has finally enhanced Transact-SQL to directly support grouped string concatenation.八年后... Microsoft SQL Server vNext 数据库引擎终于增强了 Transact-SQL 以直接支持分组字符串连接。 The Community Technical Preview version 1.0 added the STRING_AGG function and CTP 1.1 added the WITHIN GROUP clause for the STRING_AGG function. Community Technical Preview 1.0 版添加了 STRING_AGG 函数,CTP 1.1 为 STRING_AGG 函数添加了 WITHIN GROUP 子句。

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

SQL Server 2005 及更高版本允许您创建自己的自定义聚合函数,包括诸如连接之类的事情 - 请参阅链接文章底部的示例。

An example would be一个例子是

In Oracle you can use LISTAGG aggregate function.在 Oracle 中,您可以使用 LISTAGG 聚合函数。

Original records原始记录

name   type
------------
name1  type1
name2  type2
name2  type3

Sql数据库

SELECT name, LISTAGG(type, '; ') WITHIN GROUP(ORDER BY name)
FROM table
GROUP BY name

Result in导致

name   type
------------
name1  type1
name2  type2; type3

This is just an addition to Kevin Fairchild's post (very clever by the way).这只是对 Kevin Fairchild 帖子的补充(顺便说一下,这很聪明)。 I would have added it as a comment, but I don't have enough points yet :)我会把它添加为评论,但我还没有足够的分数:)

I was using this idea for a view I was working on, however the items I was concatinating contained spaces.我将这个想法用于我正在处理的视图,但是我连接的项目包含空间。 So I modified the code slightly to not use spaces as delimiters.所以我稍微修改了代码,不使用空格作为分隔符。

Again thanks for the cool workaround Kevin!再次感谢凯文的酷解决方法!

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 

This kind of question is asked here very often, and the solution is going to depend a lot on the underlying requirements:这种问题在这里经常被问到,解决方案将在很大程度上取决于底层需求:

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

and

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

Typically, there is no SQL-only way to do this without either dynamic sql, a user-defined function, or a cursor.通常,如果没有动态 sql、用户定义的函数或游标,就没有仅使用 SQL 的方法来执行此操作。

Just to add to what Cade said, this is usually a front-end display thing and should therefore be handled there.补充一下 Cade 所说的,这通常是前端显示的事情,因此应该在那里处理。 I know that sometimes it's easier to write something 100% in SQL for things like file export or other "SQL only" solutions, but most of the times this concatenation should be handled in your display layer.我知道有时在 SQL 中编写 100% 的东西更容易,例如文件导出或其他“仅限 SQL”的解决方案,但大多数情况下,这种连接应该在您的显示层中处理。

Don't need a cursor... a while loop is sufficient.不需要游标......一个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

Let's get very simple:让我们变得非常简单:

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

Replace this line:替换这一行:

select ', ' + x from (SELECT 'xxx' x union select 'yyyy') tb

With your query.随着您的查询。

You can improve performance significant the following way if group by contains mostly one item:如果 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

didn't see any cross apply answers, also no need for xml extraction.没有看到任何交叉应用答案,也不需要 xml 提取。 Here is a slightly different version of what Kevin Fairchild wrote.这是 Kevin Fairchild 所写的略有不同的版本。 It's faster and easier to use in more complex queries:在更复杂的查询中使用更快、更容易:

   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

Using the Stuff and for xml path operator to concatenate rows to string :Group By two columns -->使用 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

Using Replace Function and FOR JSON PATH使用替换函数和 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

For sample data and more ways click here有关示例数据和更多方法, 请单击此处

如果您启用了 clr,则可以使用 GitHub 中的Group_Concat

Another example without the garbage: ",TYPE).value('(./text())[1]','VARCHAR(MAX)')"另一个没有垃圾的例子: ",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

Input-output is输入输出是

*************************   ->  *********************
*   n   *   g   *   v   *       *   g   *   cg      *
*   -   *   -   *   -   *       *   -   *   -       *
*   1   *   1   *   1   *       *   1   *   1, 2    *
*   2   *   1   *   2   *       *   2   *   3       *
*   3   *   2   *   3   *       *********************
*************************   

I used this approach which may be easier to grasp.我使用了这种可能更容易掌握的方法。 Get a root element, then concat to choices any item with the same ID but not the 'official' name获取一个根元素,然后连接以选择具有相同 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 '%,%'

For all my healthcare folks out there:对于我所有的医疗保健人员:

 
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.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM