我怎样才能在Oracle中将多行组合成以逗号分隔的列表?

我有一个简单的查询:

select * from countries 

结果如下:

 country_name ------------ Albania Andorra Antigua ..... 

我想在一行中返回结果,所以像这样:

 Albania, Andorra, Antigua, ... 

当然,我可以编写一个PL / SQL函数来完成这个工作(我已经在Oracle 10g中完成了),但是对于这个任务,有没有更好的,最好是非Oracle特定的解决scheme(或者可能是内置的函数) ?

我通常会使用它来避免子查询中的多行,所以如果一个人拥有多于一个国籍,我不希望她/他在列表中重复。

我的问题是基于SQL Server 2005上的类似问题。

更新 :我的function如下所示:

 CREATE OR REPLACE FUNCTION APPEND_FIELD (sqlstr in varchar2, sep in varchar2 ) return varchar2 is ret varchar2(4000) := ''; TYPE cur_typ IS REF CURSOR; rec cur_typ; field varchar2(4000); begin OPEN rec FOR sqlstr; LOOP FETCH rec INTO field; EXIT WHEN rec%NOTFOUND; ret := ret || field || sep; END LOOP; if length(ret) = 0 then RETURN ''; else RETURN substr(ret,1,length(ret)-length(sep)); end if; end; 

这是一个简单的方法,不用创build一个函数或者创build一个函数。 它在Tom Kyte的博客中被引用。 http://tkyte.blogspot.com/2006/08/evolution.html

 create table countries ( country_name varchar2 (100)); insert into countries values ('Albania'); insert into countries values ('Andorra'); insert into countries values ('Antigua'); SELECT SUBSTR (SYS_CONNECT_BY_PATH (country_name , ','), 2) csv FROM (SELECT country_name , ROW_NUMBER () OVER (ORDER BY country_name ) rn, COUNT (*) OVER () cnt FROM countries) WHERE rn = cnt START WITH rn = 1 CONNECT BY rn = PRIOR rn + 1; CSV -------------------------- Albania,Andorra,Antigua 1 row selected. 

正如其他人所提到的,如果你使用的是11g R2或更高版本,现在可以使用listagg,这个要简单得多。

 select listagg(country_name,', ') within group(order by country_name) csv from countries; CSV -------------------------- Albania, Andorra, Antigua 1 row selected. 

WM_CONCAT函数(如果包含在数据库中,在Oracle 11.2之前)或LISTAGG (从Oracle 11.2开始)应该很好地解决这个问题。 例如,这会在您的模式中得到逗号分隔的表名称列表:

 select listagg(table_name, ', ') within group (order by table_name) from user_tables; 

要么

 select wm_concat(table_name) from user_tables; 

更多细节/选项

链接到文档

对于Oracle,您可以使用LISTAGG

你可以试试这个查询。

 select listagg(country_name,',') within group (order by country_name) cnt from countries; 

你也可以使用这个:

 SELECT RTRIM ( XMLAGG (XMLELEMENT (e, country_name || ',')).EXTRACT ('//text()'), ',') country_name FROM countries; 

使用Oracle收集function的最快方法。

你也可以这样做:

 select * 2 from ( 3 select deptno, 4 case when row_number() over (partition by deptno order by ename)=1 5 then stragg(ename) over 6 (partition by deptno 7 order by ename 8 rows between unbounded preceding 9 and unbounded following) 10 end enames 11 from emp 12 ) 13 where enames is not null 

访问网站问tom,search'stragg'或'string连接'。 很多例子。 还有一个没有logging的oracle函数来实现你的需要。

在这个例子中,我们正在创build一个函数,将不同行级别的发票保留原因的逗号分隔列表放入一个用于标题级别查询的字段中:

  FUNCTION getHoldReasonsByInvoiceId (p_InvoiceId IN NUMBER) RETURN VARCHAR2 IS v_HoldReasons VARCHAR2 (1000); v_Count NUMBER := 0; CURSOR v_HoldsCusror (p2_InvoiceId IN NUMBER) IS SELECT DISTINCT hold_reason FROM ap.AP_HOLDS_ALL APH WHERE status_flag NOT IN ('R') AND invoice_id = p2_InvoiceId; BEGIN v_HoldReasons := ' '; FOR rHR IN v_HoldsCusror (p_InvoiceId) LOOP v_Count := v_COunt + 1; IF (v_Count = 1) THEN v_HoldReasons := rHR.hold_reason; ELSE v_HoldReasons := v_HoldReasons || ', ' || rHR.hold_reason; END IF; END LOOP; RETURN v_HoldReasons; END; 

我一直不得不为此编写一些PL / SQL,或者我只是将一个','连接到该字段并复制到一个编辑器中,并从列表中删除CR,并给出单行。

那是,

 select country_name||', ' country from countries 

有一点点长途蜿蜒。

如果你看看Ask Tom,你会看到大量的可能的解决scheme,但是它们都会恢复为types声明和/或PL / SQL

问汤姆

我需要一个类似的东西,并find以下解决scheme。

 select RTRIM(XMLAGG(XMLELEMENT(e,country_name || ',')).EXTRACT('//text()'),',') country_name from 
 SELECT REPLACE(REPLACE ((SELECT TOP (100) PERCENT country_name + ', ' AS CountryName FROM country_name ORDER BY country_name FOR XML PATH('')), '&<CountryName>', ''), '&<CountryName>', '') AS CountryNames 

你可以使用这个查询来完成上述任务

 DECLARE @test NVARCHAR(max) SELECT @test = COALESCE(@test + ',', '') + field2 FROM #test SELECT field2= @test 

有关详细信息和一步一步解释,请访问以下链接
http://oops-solution.blogspot.com/2011/11/sql-server-convert-table-column-data.html