是否有可能构造SQL来连接列值 多行吗?

举例如下:

表一个

PID
A
B
C

表B

PID   SEQ    Desc

A     1      Have
A     2      a nice
A     3      day.
B     1      Nice Work.
C     1      Yes
C     2      we can 
C     3      do 
C     4      this work!

SQL的输出应该是-

PID   Desc
A     Have a nice day.
B     Nice Work.
C     Yes we can do this work!

所以基本上输出表的Desc列是来自表B的SEQ值的连接?

SQL有什么帮助吗?


当前回答

LISTAGG分析函数是在Oracle 11g Release 2中引入的,这使得聚合字符串变得非常容易。 如果你正在使用11g Release 2,你应该使用这个函数进行字符串聚合。 有关字符串连接的更多信息,请参考下面的url。

http://www.oracle-base.com/articles/misc/StringAggregationTechniques.php

字符串连接

其他回答

在想要连接的选择中,调用SQL函数。

例如:

select PID, dbo.MyConcat(PID)
   from TableA;

然后对于SQL函数:

Function MyConcat(@PID varchar(10))
returns varchar(1000)
as
begin

declare @x varchar(1000);

select @x = isnull(@x +',', @x, @x +',') + Desc
  from TableB
    where PID = @PID;

return @x;

end

Function Header语法可能是错误的,但原理是正确的。

如果必须排序,LISTAGG将提供最佳性能(00:00:05.85) SELECT pid, LISTAGG(Desc, ' ') WITHIN GROUP (ORDER BY seq) AS description 从B组pid; COLLECT在不需要排序的情况下提供最佳性能(00:00:02.90): SELECT pid, TO_STRING(CAST(COLLECT(Desc) AS varchar2_ntt)) AS val FROM B GROUP BY pid 收集与订购有点慢(00:00:07.08): SELECT pid, TO_STRING(CAST(COLLECT(Desc ORDER BY Desc) AS varchar2_ntt)) AS val FROM B GROUP BY pid

其他所有技术都要慢一些。

正如大多数答案所表明的那样,LISTAGG是显而易见的选择。然而,LISTAGG的一个恼人的方面是,如果连接的字符串的总长度超过4000个字符(SQL中VARCHAR2的限制),就会抛出下面的错误,这在12.1以上的Oracle版本中很难管理

ORA-01489:字符串连接的结果太长

12cR2中添加的一个新特性是LISTAGG的ON OVERFLOW子句。 包含这个子句的查询看起来像这样:

SELECT pid, LISTAGG(Desc, ' ' on overflow truncate) WITHIN GROUP (ORDER BY seq) AS desc
FROM B GROUP BY pid;

上面的代码将输出限制在4000个字符,但不会抛出ORA-01489错误。

下面是ON OVERFLOW子句的一些附加选项:

在溢出截断` Contd..':这将显示'Contd..“在 字符串的结尾(默认是…) ON OVERFLOW TRUNCATE ":这将显示4000个字符 没有任何终止字符串。 ON OVERFLOW TRUNCATE WITH COUNT:这将显示总数 在结束字符之后的字符数。 如:“……(5512)” ON OVERFLOW错误:如果您期望LISTAGG在 ORA-01489错误(这是默认的)。

在运行select查询之前,运行以下命令:

将服务器设置为6000大小

SELECT XMLAGG(XMLELEMENT(E,SUPLR_SUPLR_ID||',')).EXTRACT('//text()') "SUPPLIER" 
FROM SUPPLIERS;

还有一个XMLAGG函数,它适用于11.2之前的版本。因为WM_CONCAT没有文档,Oracle也不支持,所以建议不要在生产系统中使用它。

使用XMLAGG你可以做以下事情:

SELECT XMLAGG(XMLELEMENT(E,ename||',')).EXTRACT('//text()') "Result" 
FROM employee_names

它的作用是

将employee_names表中的ename列的值(用逗号连接)放在xml元素中(带有标记E) 提取其中的文本 聚合XML(连接它) 将结果列命名为Result