假如有如下表,其中各個i值對應的行數是不定的
sql> select * from t;
i a d
---------- ---------- -------------------
1 b 2008-03-27 10:55:42
1 a 2008-03-27 10:55:46
1 d 2008-03-27 10:55:30
2 z 2008-03-27 10:55:55
2 t 2008-03-27 10:55:59
要獲得如下結果,注意字串需要按照d列的時間排序:
1 d,b,a
2 z,t
這是乙個比較典型的行列轉換,有好幾種實現方法
1.自定義函式實現
create or replace function my_concat(n number)
return varchar2
istype typ_cursor is ref cursor;
v_cursor typ_cursor;
v_temp varchar2(10);
v_result varchar2(4000):= '';
v_sql varchar2(200);
begin
v_sql := 'select a from t where i=' || n ||' order by d';
open v_cursor for v_sql;
loop
fetch v_cursor into v_temp;
exit when v_cursor%notfound;
v_result := v_result ||',' || v_temp;
end loop;
return substr(v_result,2);
end;
sql> select i,my_concat(i) from t group by i;
i my_concat(i)
---------- --------------------
1 d,b,a
2 z,t
雖然這種方式可以實現需求,但是如果表t的資料量很大,i的值又很多的情況下,因為針對每個i值都要執行一句select,掃瞄和排序的次數和i的值成正比,效能會非常差。
2.使用sys_connect_by_path
select i,ltrim(max(sys_connect_by_path(a,',')),',') a
from
(select i,a,d,min(d) over(partition by i) d_min,
(row_number() over(order by i,d))+(dense_rank() over (order by i)) numid
from t
)start with d=d_min connect by numid-1=prior numid
group by i;
從執行計畫上來看,這種方式只需要掃瞄兩次表,比自定義函式的方法,效率要高很多,尤其是表中資料量較大的時候:
3.使用wm_sys.wm_concat
這個函式也可以實現類似的行列轉換需求,但是似乎沒有辦法做到直接根據另外一列排序,所以需要先通過子查詢或者臨時表排好序
sql> select i,wmsys.wm_concat(a) from t group by i;
i wmsys.wm_concat(a)
---------- --------------------
1 b,a,d
2 z,t
sql> select i,wmsys.wm_concat(a)
2 from
3 (select * from t order by i,d)
4 group by i;
i wmsys.wm_concat(a)
---------- --------------------
1 d,b,a
2 z,t
執行計畫上看,只需要做一次表掃瞄就可以了,但是這個函式是加密過的,執行計畫並不能顯示函式內部的操作。
Oracle行列轉換
1.列轉行 create table t col row id int,c1 varchar2 10 c2 varchar2 10 c3 varchar2 10 insert into t col row values 1,v11 v21 v31 insert into t col row valu...
Oracle行列轉換
行轉列 select count over cnt,rn,str from select rownum rn,substr a.str,instr a.str,1,a.n 1,instr a.str,1,a.n 1 instr a.str,1,a.n 1 str from select a,b,c,...
oracle 行列轉換
q 如何實現行列轉換 a 1 固定列數的行列轉換 如student subject grade student1 語文 80 student1 數學 70 student1 英語 60 student2 語文 90 student2 數學 80 student2 英語 100 轉換為 語文 數學 英...