http://lovejuan1314.iteye.com/blog/413694

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

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

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
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.自定义函数实现

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
create or replace function my_concat(n number)
return varchar2
is
type 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

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;
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代码  
  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
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

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

其他一些方法:
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

Oracle 多行合并一行 方法相关推荐

  1. ORACLE纯SQL实现多行合并一行

    ORACLE纯SQL实现多行合并一行 项目中遇到一个需求,需要将多行合并为一行. 表结构如下: NAME                            Null           Type ...

  2. oracle一行变一列,oracle多行变一行

    今天遇到将多行转为一行的一个操作,多谢oracle开发板的 wildwave 提供了比较通用的解决办法,同时也将自己搜到的这方面资料整理如下,多是用于连接列值的. String集聚连接技术 需要将多行 ...

  3. element-ui table中span-method(行合并)方法使用

    [官方文档span-method方法](https://element.eleme.cn/#/zh-CN/component/table#table-attributes)中给的例子是第一列每两行进行 ...

  4. oracle 多行合并成一行

    假如有如下表,其中各个i值对应的行数是不定的 Sql代码   SQL> select * from t; I A          D ---------- ---------- ------- ...

  5. Oracle的 listagg 函数,多行合并一行

    转自:https://blog.csdn.net/Huay_Li/article/details/81257655 --创建学校类型-年级表 create table SchoolGrade(scho ...

  6. easyui一行显示多行_easyui datagrid以及oracle中的多行合并一行

    页面显示: 前端代码: toolbar="#toolbar" rownumbers="true" singleSelect="true" f ...

  7. pandas多行合并一行_Pandas函数妙用

    ❝ 文章来源:towardsdatascience 作者:Soner Yıldırım 翻译\编辑:Python大数据分析 ❞ 看标题是否似曾相似?之前为大家介绍过10个高效的pandas函数,颇受欢 ...

  8. Oracle多行合并为单行多列

    表结构如上,因为 相同,现在需要把CONTAINER_CODE和BILL_CODE相同的行,的到达时间合并为自定义的列 实现类似这样的效果: Oracle实现的思路是:对这两个字段group by分组 ...

  9. mysql和sqlserver共存_mysql 和sqlserver的 多行合并成一行

    mysql 和sqlserver的 多行合并成一行 mysql  多行合并: mysql 内置函数 group_concat(exper  SEPARATOR " " ) expe ...

最新文章

  1. ZStack--通过Ansible实现全自动化
  2. 【 FPGA 】抢占式优先级译码器电路
  3. linux下rsync+inotify实现服务器之间文件实时同步
  4. python字典内存分析_Python减少字典对象占用的七成内存
  5. virsh KVM管理工具
  6. java nio 堆外内存_Java堆外内存之突破JVM枷锁
  7. 安全公司笔试面试题总结
  8. java语言情话_程序员的浪漫:用 java 实现每天给对象法发情话
  9. mysql固定某列获取不连续的值_SQL-怎么把一列不规律的值,取出其中连续段的首尾数字?...
  10. shell 整理(36)===写斐波那契数列
  11. linux字符界面播放vcd,在Linux中制作VCD
  12. 职场中的你有危机感吗?在职提升来社科院与杜兰大学金融管理硕士项目
  13. 【全】可供选择的软件开源协议的罗列
  14. 托福高频真词List05 // 附托福TPO阅读真题
  15. 改造,分刷新显卡BIOS和驱动修改两种
  16. 半导体物理·笔记之④晶体中原子的电子状态——能带
  17. ubuntu 20.04 调节屏幕亮度
  18. halcon的长度和角度测量
  19. js怎么实现倒计时效果
  20. JavaScript在函数中定义全局变量

热门文章

  1. 《数据库原理与应用》(第三版)第15章 备份和恢复数据库 基础 习题参考答案
  2. 信息学奥赛一本通(C++)在线评测系统——基础(一)C++语言—— 1123:图像相似度
  3. 深度学习——05、深度学习框架Caffe
  4. VS2017断点调试
  5. 【C++】 C++标准模板库(八) MultiMaps
  6. 大脑构造图与功能解析_解析地轨、隐藏轨推拉门及折叠门的构造做法,收藏学习...
  7. android点九,android关于点九(.9)图片
  8. Win Api 有名管道通信--扫雷辅助(四)
  9. Qt程序打包发布方法(使用官方提供的windeployqt工具)
  10. 计算机除法和取余在实际运用中的意义