Oracle 多行合并一行 方法

引用自:http://lovejuan1314.iteye.com/blog/413694


OTE:特别声明一下内容转自网络 
http://www.ningoo.net/html/2008/how_to_do_string_aggregate_on_oracle.html 

假如有如下表,其中各个i值对应的行数是不定的 

Sql代码   收藏代码
  1. SQL> select * from t;  
  2.   
  3.          I A          D  
  4. ---------- ---------- -------------------  
  5.          1 b          2008-03-27 10:55:42  
  6.          1 a          2008-03-27 10:55:46  
  7.          1 d          2008-03-27 10:55:30  
  8.          2 z          2008-03-27 10:55:55  
  9.          2 t          2008-03-27 10:55:59  
  10.   
  11. --- 要获得如下结果,注意字符串需要按照D列的时间排序:  
  12.   
  13. 1  d,b,a  
  14. 2  z,t  


这是一个比较典型的行列转换,有好几种实现方法 

1.自定义函数实现  

Sql代码   收藏代码
  1. create or replace function my_concat(n number)  
  2. return varchar2  
  3. is  
  4.  type typ_cursor is ref cursor;  
  5.  v_cursor typ_cursor;  
  6.  v_temp varchar2(10);  
  7.  v_result varchar2(4000):= '';  
  8.  v_sql varchar2(200);  
  9. begin  
  10.  v_sql := 'select a from t where i=' || n ||' order by d';  
  11.  open v_cursor for v_sql;  
  12.  loop  
  13.     fetch v_cursor into v_temp;  
  14.     exit when v_cursor%notfound;  
  15.     v_result := v_result ||',' || v_temp;  
  16.  end loop;  
  17.  return substr(v_result,2);  
  18. end;  
  19.   
  20. SQL> select i,my_concat(i) from t group by i;  
  21.   
  22.          I MY_CONCAT(I)  
  23. ---------- --------------------  
  24.          1 d,b,a  
  25.          2 z,t  


虽然这种方式可以实现需求,但是如果表t的数据量很大,i的值又很多的情况下,因为针对每个i值都要执行一句select,扫描和排序的次数和i的值成正比,性能会非常差。 

2.使用sys_connect_by_path  

Sql代码   收藏代码
  1. select i,ltrim(max(sys_connect_by_path(a,',')),',') a  
  2. from  
  3. (  
  4. select i,a,d,min(d) over(partition by i) d_min,  
  5. (row_number() over(order by i,d))+(dense_rank() over (order by i)) numid  
  6. from t  
  7. )  
  8. start with d=d_min connect by numid-1=prior numid  
  9. group by i;  

从执行计划上来看,这种方式只需要扫描两次表,比自定义函数的方法,效率要高很多,尤其是表中数据量较大的时候: 
 
3.使用wm_sys.wm_concat  
这个函数也可以实现类似的行列转换需求,但是似乎没有办法做到直接根据另外一列排序,所以需要先通过子查询或者临时表排好序: 
Sql代码   收藏代码
  1. SQL> select i,wmsys.wm_concat(a) from t group by i;  
  2.   
  3.          I WMSYS.WM_CONCAT(A)  
  4. ---------- --------------------  
  5.          1 b,a,d  
  6.          2 z,t  
  7.   
  8. SQL> select i,wmsys.wm_concat(a)  
  9.   2  from  
  10.   3  (select * from t order by i,d)  
  11.   4  group by i;  
  12.   
  13.          I WMSYS.WM_CONCAT(A)  
  14. ---------- --------------------  
  15.          1 d,b,a  
  16.          2 z,t  

执行计划上看,只需要做一次表扫描就可以了,但是这个函数是加密过的,执行计划并不能显示函数内部的操作。
 

其他一些方法: 
http://asktom.oracle.com/pls/asktom/f?p=100:11:0::::P11_QUESTION_ID:2196162600402  
http://asktom.oracle.com/pls/asktom/f?p=100:11:0::::P11_QUESTION_ID:15637744429336

<think>嗯,用户问的是如何在Oracle数据库中将多行数据合并一行,使用LISTAGG函数的示例。我需要先回忆一下LISTAGG函数的用法和相关注意事项。根据提供的引用内容,比如引用1、3、4都提到了LISTAGG函数的基本用法,应该是用LISTAGG(字段, 分隔符) WITHIN GROUP (ORDER BY 排序字段)然后配合GROUP BY分组。 首先,我得确认用户是否已经了解这个函数的基本结构,可能需要先给出一个简单的示例,比如合并员工姓名为逗号分隔的字符串。然后,可能需要提到分组的情况,比如按部门合并,这时候需要用GROUP BY子句,如引用2和引用3提到的。 另外,用户可能需要知道如何处理可能的问题,比如字符串长度超限的情况,这时候引用2提到了缺点,结果字符串长度有限制,可能需要考虑替代方案,比如WM_CONCAT或者XMLAGG函数,虽然这些可能在不同版本中有兼容性问题。例如,引用4提到了xmlagg函数,但用户的问题主要围绕LISTAGG,所以可能需要简要提到替代方法。 还需要注意语法是否正确,比如WITHIN GROUP后面的ORDER BY是否正确,分隔符的使用是否正确。同时,可能需要提醒用户关于性能问题,尤其是在大数据量或复杂查询时,如引用2中的缺点部分。 另外,用户可能需要具体的示例代码,比如如何将PRS_CALC_DATA表中的emp_uid按逗号分隔合并,如引用4中的示例。应该给出这样的例子,并说明各部分的作用,比如WITHIN GROUP和ORDER BY的必要性。 最后,可能需要总结优缺点,帮助用户决定是否适合使用LISTAGG,或者是否需要考虑其他方法。同时,生相关问题的时候,可以涉及常见错误、处理超长字符串的方法,或者与其他函数的比较,如引用提到的XMLAGG。</think>在Oracle数据库中,使用$LISTAGG$函数可以高效地将多行数据合并一行。以下是具体实现方法和示例: ### 一、$LISTAGG$函数语法 $$ \text{LISTAGG}(column\_name, delimiter) \text{ WITHIN GROUP } (\text{ORDER BY } sort\_column) $$ - **$column\_name$**: 需要合并的字段 - **$delimiter$**: 分隔符(如逗号、分号) - **$sort\_column$**: 合并结果的排序依据字段 ### 二、基础示例 假设表$employee$包含部门编号$dept_id$和员工姓名$name$: ```sql SELECT dept_id, LISTAGG(name, ', ') WITHIN GROUP (ORDER BY name) AS employees FROM employee GROUP BY dept_id; ``` **结果示例**: ``` DEPT_ID | EMPLOYEES --------|------------------- 10 | Alice, Bob 20 | Charlie, David ``` 通过$GROUP BY$按部门分组,将同部门员工姓名合并为逗号分隔的字符串,并按姓名排序[^3]。 ### 三、进阶用法 1. **无分组合并所有行**: ```sql SELECT LISTAGG(zjxm, ';') WITHIN GROUP (ORDER BY zjxm) AS merged_zjxm FROM pb_zjzcy; ``` 将所有$zjxm$字段合并为分号分隔的字符串,并按原字段排序[^1]。 2. **处理重复值**: 添加$DISTINCT$关键字去重: ```sql SELECT LISTAGG(DISTINCT emp_uid, ',') WITHIN GROUP (ORDER BY emp_uid) FROM PRS_CALC_DATA; ``` ### 四、注意事项 1. **字符串长度限制**: $LISTAGG$结果长度超过4000字节会报错。解决方法是使用子查询分组合并,或改用$XMLAGG$函数: ```sql SELECT RTRIM(XMLAGG(XMLELEMENT(e, emp_uid || ',')).EXTRACT('//text()'), ',') FROM PRS_CALC_DATA; ``` [^4] 2. **性能优化**: - 避免在大数据集直接使用,可通过缩小$WHERE$条件范围 - 若需处理多列合并,建议分阶段操作
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值