我该如何获得:
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
STRING_AGG
:SELECT id, STRING_AGG(CONCAT(name, ':', [value]), ', ')
FROM #YourTable
GROUP BY id
使用XML路径拼接文本时,并不能像您期望的那样完美地连接...它会将"&"替换为"&",并且还会破坏<
和>
等标签。可能还有其他一些问题,不确定...但您可以尝试这个方法:
我发现了一个解决方法...你需要将以下内容替换:
FOR XML PATH('')
)
使用:
FOR XML PATH(''),TYPE
).value('(./text())[1]','VARCHAR(MAX)')
如果您正在使用NVARCHAR(MAX)
,则可以使用FOR XML PATH('')
进行字符串连接。
为什么SQL
没有连接聚合函数?这真是太让人头疼了。
当我尝试将Kevin Fairchild的建议应用于包含空格和特殊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
删除前两个字符。
通过使用TYPE指令自动处理XML编码。
另一种选择是使用 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
从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;
这只是对Kevin Fairchild的帖子的补充(非常聪明)。我本来想将其作为评论添加,但我还没有足够的积分 :)
我正在使用这个想法来处理一个视图,但是我要连接的项包含空格。因此,我稍微修改了代码,不使用空格作为分隔符。
再次感谢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
https://stackoverflow.com/search?q=sql+pivot
并且
通常情况下,没有仅使用SQL的方法来完成此操作,需要使用动态SQL、用户定义函数或游标。
GROUP_CONCAT()
聚合函数轻松解决这种问题,但在Microsoft SQL Server上解决这个问题则比较麻烦。需要帮助的话可以参考以下Stack Overflow的问题: "如何基于关系获取一个记录的多个记录?" - Bill Karwin