我可以用逗号将多行分隔为一列吗? [复制]
Posted
技术标签:
【中文标题】我可以用逗号将多行分隔为一列吗? [复制]【英文标题】:Can I Comma Delimit Multiple Rows Into One Column? [duplicate] 【发布时间】:2011-01-03 23:59:57 【问题描述】:我正在尝试在我的 SQL Server 数据库中合并类似的内容:
[TicketID], [Person] T0001 爱丽丝 T0001 鲍勃 T0002 凯瑟琳 T0002 道格 T0003 伊莱恩
进入这个:
[TicketID]、[人物] T0001 爱丽丝,鲍勃 T0002 凯瑟琳,道格 T0003 伊莱恩
我需要在 SQL Server 和 Oracle 中都这样做。
我找到了用于 mysql 的函数 GROUP_CONCAT
,它完全可以满足我的需要,但这里不能选择 MySQL。
编辑:测试台:
DECLARE @Tickets TABLE (
[TicketID] char(5) NOT NULL,
[Person] nvarchar(15) NOT NULL
)
INSERT INTO @Tickets VALUES
('T0001', 'Alice'),
('T0001', 'Bob'),
('T0002', 'Catherine'),
('T0002', 'Doug'),
('T0003', 'Elaine')
SELECT * FROM @Tickets
【问题讨论】:
【参考方案1】:我已经找到了在 Oracle 中执行此操作的方法,但我仍需要在 SQL Server 中执行此操作。
来自http://technology.amis.nl/blog/6118/oracle-rdbms-11gr2-listagg-new-aggregation-operator-for-creating-comma-delimited-strings (感谢tanging)(ORACLE 11 及更高版本)
select
TicketId,
listagg(Person, ', ') People
from
table
group by
TicketId
发件人:http://halisway.blogspot.com/2006/08/oracle-groupconcat-updated-again.html
with
data
as
(
select
TicketId,
Person,
ROW_NUMBER() over (partition by TicketId order by Person) "rownum",
COUNT(*) over (partition by TicketId) "count"
from
Table
)
select
TicketId,
LTRIM(sys_connect_by_path(Person,','),',') People
from
data
where
"rownum" = "count"
start with
"rownum" = 1
connect by
prior TicketId = TicketId
and
prior "rownum" = "rownum" - 1
order by
TicketId
【讨论】:
+1:感谢 Oracle 示例。在 Oracle 中有几种方法可以做到这一点,但有些方法包括使用不受支持的功能。 我知道这是旧的,但如果你使用的是 11g,你有 ListAgg(这似乎与 MySQL Group_CONCAT 非常相似):technology.amis.nl/blog/6118/… & download.oracle.com/docs/cd/E11882_01/server.112/e10592/… @tanging:太棒了!我正在编辑我的回复以反映这一点! LISTAGG 文档的正确链接docs.oracle.com/cd/E11882_01/server.112/e41084/functions089.htm【参考方案2】:这是一个适用于 SQL Server 2005+ 的解决方案:
SELECT t.TicketID,
STUFF(ISNULL((SELECT ', ' + x.Person
FROM @Tickets x
WHERE x.TicketID = t.TicketID
GROUP BY x.Person
FOR XML PATH (''), TYPE).value('.','VARCHAR(max)'), ''), 1, 2, '') [No Preceeding Comma],
ISNULL((SELECT ', ' + x.Person
FROM @Tickets x
WHERE x.TicketID = t.TicketID
GROUP BY x.Person
FOR XML PATH (''), TYPE).value('.','VARCHAR(max)'), '') [Preceeding Comma If Not Empty]
FROM @Tickets t
GROUP BY t.TicketID
参考:
STUFF (Transact-SQL)【讨论】:
这不起作用...您所做的分组没有使用人员字段,因此它失败并且xml路径不能将逗号(,)作为标识符,因为它无法从中创建元素.. 坏蛋!你在底部缺少一个分组,但太棒了! 我很好奇:这是在服务器上使用查询运行还是在内存中连接结果更快? 游标是必要的,据我所知,它们非常慢。 不管怎样,我过去也有类似的需求,并尝试了许多不同的方法,包括编写我自己的 CLR .NET 聚合函数。到目前为止,OMG Ponies 展示的 XPATH 解决方案是最快的。【参考方案3】:为了完整起见,还有 MySQL 版本:
select
TicketId,
GROUP_CONCAT(Person ORDER BY Person SEPARATOR ', ') People
from
table
group by
TicketId
【讨论】:
【参考方案4】:一个例子
SELECT DISTINCT
t.TicketID,
STUFF((SELECT ', ', i.Person as [text()]
FROM @Tickets i
WHERE i.TicketID = t.TicketID
FOR XML PATH ('')), 1, 2, '') as People
FROM
@Tickets t
............或尝试............
SELECT DISTINCT
t.TicketID,
STUFF((SELECT ', ' + i.Person /* notice this line is different */
FROM @Tickets i
WHERE i.TicketID = t.TicketID
FOR XML PATH ('')), 1, 2, '') as People
FROM
@Tickets t
/* 当我将它用于我的桌子时,这很有效,并且归功于我的经理,ROCKS! */
【讨论】:
这会留下一个逗号。 我冒昧地将此匹配作为示例。这比以前的最佳答案执行得快得多。 @JohnGietzen 性能提升并非没有代价。对于包含例如&
的值,这将无法按预期工作。
使用disntinct
也不是最好的方法,如果你用大量数据进行测试,你会发现group by
解决方案更快。
看看 cmets 到这个关于不同的答案。 ***.com/questions/9811577/…【参考方案5】:
DECLARE @Tickets TABLE (
[TicketID] char(5) NOT NULL,
[Person] nvarchar(15) NOT NULL
)
INSERT INTO @Tickets VALUES
('T0001', 'Alice'),
('T0001', 'Bob'),
('T0002', 'Catherine'),
('T0002', 'Doug'),
('T0003', 'Elaine')
SELECT * FROM @Tickets
Select [TicketID],
STUFF((SELECT ',' + Person FROM @Tickets WHERE (
TicketID=Result.TicketID) FOR XML PATH ('')),1,1,'') AS BATCHNOLIST
From @Tickets AS Result
GROUP BY TicketID
【讨论】:
在我的场景中,“Person”的类型是 Money,并且这种解决方案被证明比“OMG Ponies”解决方案更快、更准确。 确认它完美运行 - thx 一百万!以上是关于我可以用逗号将多行分隔为一列吗? [复制]的主要内容,如果未能解决你的问题,请参考以下文章