【问题标题】: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

【问题讨论】:

    标签: sql sql-server sql-server-2008 tsql


    【解决方案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
    

    【讨论】:

    【解决方案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
    

    参考:

    【讨论】:

    • 这不起作用...您所做的分组没有使用人员字段,因此它失败并且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 到这个关于不同的答案。 stackoverflow.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 一百万!
      猜你喜欢
      • 1970-01-01
      • 2018-11-02
      • 2010-10-27
      • 1970-01-01
      • 1970-01-01
      • 2011-04-25
      • 2019-11-10
      • 1970-01-01
      相关资源
      最近更新 更多