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


373

どうすれば取得できますか:

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

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

18
この種の問題は、GROUP_CONCAT()集計関数を使用してMySQLで簡単に解決できますが、Microsoft SQL Serverで解決するのはより厄介です。ヘルプについては、次のSOの質問を参照してください:「リレーションに基づいて1つのレコードに対して複数のレコードを取得する方法
Bill Karwin

1
マイクロソフトアカウントを持つ誰もが、接続上の簡単な解決策のために投票する必要がありますconnect.microsoft.com/SQLServer/feedback/details/427987/...
イェンスMühlenhoffに

1
あなたは、T-SQLが強化されるまでの代替として、ここで見つけSQLCLR集約を使用することができます。groupconcat.codeplex.com
オーランドColamatteo

回答:


550

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

6
なぜ一時テーブルをロックしないのですか?
Amy B

3
これは、私が今まで見た中で最もクールなSQLです。大規模なデータセットに対して「高速」であるかどうかを考えますか?カーソルのようにクロールを開始しませんか?もっと多くの人がこの狂気に投票してほしいです。
user12861 2008年

6
ええ 私はそれのサブクエリスタイルが嫌いです。JOINSはとても良いです。このソリューションでそれを利用できるとは思わないでください。とにかく、私とは別に、このようなことを学ぶのが好きなSQLドックが他にもあるのを見てうれしいです。皆さんへの称賛:)
ケビンフェアチャイルド

6
文字列操作を行う少しクリーンな方法:STUFF((SELECT '、' + [Name] + ':' + CAST([Value] AS VARCHAR(MAX))FROM #YourTable WHERE(ID = Results.ID)FOR XML PATH( ''))、1、2、 '')AS NameValues
Jonathan Sayce

3
私が見つけたものに注意してください。大文字と小文字を区別しない環境でも、クエリの.valueの部分は小文字にする必要があります。これはXMLであるためだと思います。これは大文字と小文字が区別されます
Jaloopa

136

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

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

完璧に動作します!
argoo

1
これはうまくいき、受け入れられた答えよりも優れています。
Jannick Breunis

51

XMLパスを使用すると、期待どおりに完全に連結されません... "&"を "&amp;"に置き換えます また、<" and "> ...他にもいくつか問題があるかもしれませんが、わかりません...しかし、これを試すことができます

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

FOR XML PATH('')
)

と:

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

...またはNVARCHAR(MAX)、あなたが使用している場合。

なぜ地獄にSQL連結集約関数がないのですか?これはPITAです。


2
出力をエンコードしないための最良の方法を探してネットを精査しました。どうもありがとうございます!これが決定的な答えです。CONCAT()集約関数のように、MSがこれに対する適切なサポートを追加するまでです。私が行うことは、これをOuter-Applyに入れて、連結したフィールドを返すことです。私は自分の選択ステートメントにネストされた選択を追加するのが好きではありません。
MikeTeeVee 2013年

Valueを使用しないと、テキストがXMLエンコードされた文字であるという問題が発生する可能性があることに同意しました。SQL Serverでグループ化された連結のシナリオをカバーする私のブログを見つけてください。 blog.vcillusion.co.in/...
vCillusion

40

私はスペースや特殊な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. &gt; &lt; etc*/
    ).value('.','VARCHAR(MAX)') 
  ,1,2,'') as NameValues
FROM    #YourTable Results
GROUP BY ID

DROP TABLE #YourTable

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

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


21

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サーバーの問題を解決するには、常にCTEと再帰CTEを使用することを好みます。これは私にとって素晴らしい作品です!
gbdavid 2015年

外部適用のクエリでそれを使用することは可能ですか?
ホールで発砲

14

SQLCLRアグリゲートをhttp://groupconcat.codeplex.comからインストールします。

次に、次のようなコードを記述して、要求した結果を取得できます。

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;

数年前に使用しましたが、構文はすべての「XMLパス」のトリックよりもはるかにクリーンで、非常にうまく機能します。SQL CLR関数がオプションの場合は、これを強くお勧めします。
AFract

12

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


4
残念ながら、これにはCLRアセンブリの使用(?)が必要です。これは、対処する別の問題です:-/

1
例では、実際の連結の実装にCLRを使用していますが、これは必須ではありません。連結集約関数でFOR XMLを使用するようにすると、少なくとも将来的にそれを呼び出すのが簡単になります!
Shiv 2016

12

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/en-us/library/mt775028.aspx


9

これは、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 

9

例は

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

6
見た目はいいですが、質問は特にOracleに関するものではありません。
user12861 2013

13
わかります。しかし、私はOracleについても同じことを探していたので、私のような他の人のためにここに置いておきたいと思いました:)
Michal B.

@MichalB。あなたはwithin構文を見逃していませんか?例:group(order by name)内のlistagg(type、 '、')?
グレゴリー2017

@gregory:私は自分の回答を編集しました。私の古いソリューションは昔はうまく機能していたと思います。あなたが提案した現在のフォームは確かに機能します、ありがとう。
Michal

1
将来の人々のために-あなたは異なるプラットフォームのような重要な違いのためにあなた自身の答えで新しい質問を書くことができます
Mike M

7

この種の質問はここで頻繁に尋ねられ、解決策は根本的な要件に大きく依存します。

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

そして

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

通常、動的SQL、ユーザー定義関数、またはカーソルなしでこれを行うSQL専用の方法はありません。


2
違います。cte:sを使用するcyberkiwiのソリューションは、ベンダー固有のハッカーのない純粋なSQLです。
ビョルンリンドクビスト2013

1
質問と回答の時点では、再帰的なCTEをそれほど移植性があるとは見なしていませんでしたが、現在、Oracleによってサポートされています。最善の解決策はプラットフォームに依存することになります。SQL Serverの場合は、FOR XML手法または顧客のCLR集計が最も可能性が高いです。
Cade Roux 2013

1
すべての質問に対する究極の答えは? stackoverflow.com/search?q= [質問は
何でも

7

Cadeの発言に加えて、これは通常はフロントエンドの表示であるため、そこで処理する必要があります。ファイルエクスポートやその他の「SQLのみ」のソリューションなどの場合、SQLで何かを100%書く方が簡単な場合がありますが、ほとんどの場合、この連結はディスプレイレイヤーで処理する必要があります。


11
グループ化は現在、フロントエンド表示のものですか?グループ化された結果セットの1つの列を連結するための有効なシナリオはたくさんあります。
MGOwen 2016

5

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


@marc_sおそらくより良い批判は、PRIMARY KEYがテーブル変数で宣言されるべきであるということです。
エイミーB

@marc_s詳細な調査では、その記事は偽物です-IO測定なしのパフォーマンスに関するほとんどすべての議論です。私はLAGについて学びました-それでありがとう。
エイミーB

4

非常に簡単にしましょう:

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

あなたのクエリで。


3

クロス適用の回答はありませんでした。また、xml抽出の必要もありませんでした。これは、ケビンフェアチャイルドが書いたものとは少し異なるバージョンです。より複雑なクエリで使用する方が速くて簡単です。

   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

1
Valueを使用しないと、テキストがXMLエンコードされた文字であるという問題が発生する可能性があります
vCillusion

2

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

リストに重複した名前を付けたくない場合は、名前の重複を避けます。
jnm2 2016年

1

Replace関数と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

サンプルデータとその他の方法については、ここをクリックしてください


弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.