web-dev-qa-db-ja.com

SQL ServerでGROUP BYを使用して文字列を連結する方法

どうやって手に入れますか:

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

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

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
511
Kevin Fairchild

SQL Server 2017またはSQL Server Vnextの場合、SQL Azureでは以下のようにstring_aggを使用できます。

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

xMLパスを使用しても、予想どおりに完全には連結されません。 "&"が "&amp;"に置き換えられます。そして<" and ">を台無しにするかもしれません…多分他のいくつかのこと、確かではありません…しかしあなたはこれを試すことができます

私はこれのための回避策に出会いました...あなたは置き換える必要があります:

FOR XML PATH('')
)

と:

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

...またはNVARCHAR(MAX)であれば、それを使用しています。

なぜSQLが連結集約関数を持たないのですか?これはピタです。

49
Allen

Kevin Fairchildの提案をエンコードしたスペースと特殊なXML文字(&<>)を含む文字列を処理するように変換しようとしたとき、2つの問題に遭遇しました。

私のコードの最終版(これは元の質問には答えませんが、誰かに役立つかもしれません)は次のようになります。

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

スペースを区切り文字として使用してすべてのスペースをコンマで置き換えるのではなく、各値の前にコンマとスペースを追加してから、最初の2文字を削除するためにSTUFFを使用します。

XMLエンコーディングは、 TYPE ディレクティブを使用して自動的に処理されます。

38
Jonathan Sayce

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
21
cyberkiwi

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;
13

SQL Server 2005以降では、連結などを含む独自の カスタム集計関数 を作成できます。リンク先記事の下部にあるサンプルを参照してください。

12
Joel Coehoorn

8年後... Microsoft SQL Server vNextデータベースエンジンは、グループ化された文字列連結を直接サポートするように、やっとTransact-SQLを強化しました。 Community Technical Previewバージョン1.0ではSTRING_AGG関数が追加され、CTP 1.1ではSTRING_AGG関数用のWITHIN GROUP句が追加されました。

参照: https://msdn.Microsoft.com/ja-jp/library/mt775028.aspx

10
Shem Sargent

例は

Oracleでは、LISTAGG集計関数を使用できます。

元のレコード

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

SQL

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

の結果

name   type
------------
name1  type1
name2  type2; type3
9
Michal B.

この種の質問はここで非常に頻繁に尋ねられ、解決策は根本的な要件によって大きく左右されます。

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

そして

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

通常、動的SQL、ユーザー定義関数、またはカーソルを使用せずにこれを実行するためのSQLのみの方法はありません。

7
Cade Roux

Cadeが言ったことに加えて、これは通常フロントエンドの表示物であり、したがってそこで扱われるべきです。私は、ファイルのエクスポートやその他の「SQLのみ」の解決策などのために、100%SQLで何かを書くのが簡単であることを知っていますが、ほとんどの場合、この連結はあなたの表示層で扱うべきです。

7
Tom H

これは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 
6
Phillip

カーソルは必要ありません。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
5
Amy B

とても簡単にしましょう。

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

あなたの質問で。

4
Marquinho Peli

クロスアプライの回答も見られず、また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
3
Mordechai

Group byにほとんど1つの項目が含まれている場合は、以下の方法でパフォーマンスを大幅に向上させることができます。

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
2
Eduard

置換関数と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

サンプルデータおよびその他の方法については、 ここをクリック

1
Mahesh