1. 程式人生 > >Oracle 多行合併一行 方法

Oracle 多行合併一行 方法

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

Sql程式碼  收藏程式碼
  1. SQL> select * from t;  
  2.          I A          D  
  3. ---------- ---------- -------------------  
  4.          1 b          2008-03-27 10:55:42  
  5.          1 a          2008-03-27 10:55:46  
  6.          1 d          2008-03-27 10:55:30  
  7.          2 z          2008-03-27 10:55:55  
  8.          2 t          2008-03-27 10:55:59  
  9. --- 要獲得如下結果,注意字串需要按照D列的時間排序:  
  10. 1  d,b,a  
  11. 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. SQL> select
     i,my_concat(i) from t group by i;  
  20.          I MY_CONCAT(I)  
  21. ---------- --------------------  
  22.          1 d,b,a  
  23.          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.          I WMSYS.WM_CONCAT(A)  
  3. ---------- --------------------  
  4.          1 b,a,d  
  5.          2 z,t  
  6. SQL> select i,wmsys.wm_concat(a)  
  7.   2  from  
  8.   3  (select * from t order by i,d)  
  9.   4  group by i;  
  10.          I WMSYS.WM_CONCAT(A)  
  11. ---------- --------------------  
  12.          1 d,b,a  
  13.          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