どうすれば取得できますか:
id Name Value
1 A 4
1 B 8
2 C 9
に
id Column
1 A:4, B:8
2 C:9
どうすれば取得できますか:
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パスを使用すると、期待どおりに完全に連結されません... "&"を "&amp;"に置き換えます また、<" and ">
...他にもいくつか問題があるかもしれませんが、わかりません...しかし、これを試すことができます
私はこれの回避策に出くわしました...あなたは置き換える必要があります:
FOR XML PATH('')
)
と:
FOR XML PATH(''),TYPE
).value('(./text())[1]','VARCHAR(MAX)')
...またはNVARCHAR(MAX)
、あなたが使用している場合。
なぜ地獄にSQL
連結集約関数がないのですか?これはPITAです。
私はスペースや特殊な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
、最初の2文字を削除するために使用します。
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
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;
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
これは、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
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
この種の質問はここで頻繁に尋ねられ、解決策は根本的な要件に大きく依存します。
https://stackoverflow.com/search?q=sql+pivot
そして
https://stackoverflow.com/search?q=sql+concatenate
通常、動的SQL、ユーザー定義関数、またはカーソルなしでこれを行うSQL専用の方法はありません。
Cadeの発言に加えて、これは通常はフロントエンドの表示であるため、そこで処理する必要があります。ファイルエクスポートやその他の「SQLのみ」のソリューションなどの場合、SQLで何かを100%書く方が簡単な場合がありますが、ほとんどの場合、この連結はディスプレイレイヤーで処理する必要があります。
カーソルは必要ありません... 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
クロス適用の回答はありませんでした。また、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
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
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
サンプルデータとその他の方法については、ここをクリックしてください
clrを有効にしている場合は、GitHubのGroup_Concatライブラリを使用できます。
GROUP_CONCAT()
集計関数を使用してMySQLで簡単に解決できますが、Microsoft SQL Serverで解決するのはより厄介です。ヘルプについては、次のSOの質問を参照してください:「リレーションに基づいて1つのレコードに対して複数のレコードを取得する方法」