oracle listagg函数、lag函数、lead函数 实例

Oracle大师Thomas Kyte在他的经典著作中,反复强调过一个实现需求方案选取顺序:

“如果你可以使用一句SQL解决的需求,就使用一句SQL;如果不可以,就考虑PL/SQL是否可以;如果PL/SQL实现不了,就考虑Java存储过程是否可以;如果这些都不可能实现,那么就需要考虑你是否真的需要实现这个需求。”

各个关系型DBMS产品都在遵守关系型数据库模型的基本体系架构,遵循通用的SQL国际规范。同时,为了更好地配合自身数据库实现的特征,以及提供更加丰富的功能,各个DBMS纷纷在标准SQL上进行了一系列语法、语义以及功能上的拓展实现。Oracle从商用化开始,就不断的在这条道路上不断探索。借助Oracle提供的很多拓展SQL语句和函数,可以帮助我们将过去很复杂的需求,通过简单的一句SQL加以实现。

本篇就介绍三个日常开发中有用的函数方法。

1、listagg

我们有时候会遇到这样的需求:“对员工列表进行操作,将每个部门的员工名称横向排列,以逗号进行分割”。

员工表我们使用scott用户schema下的emp表。

SQL> select * from emp;

EMPNO ENAME     JOB        MGR HIREDATE         SAL     COMM DEPTNO

----- ---------- --------- ----- ----------- --------- --------- ------

7369 SMITH     CLERK     7902 1980-12-17    800.00              20

7876 ADAMS     CLERK     7788 1987-5-23    1100.00              20

7900 JAMES     CLERK     7698 1981-12-3     950.00              30

7902 FORD      ANALYST   7566 1981-12-3    3000.00              20

7934 MILLER    CLERK     7782 1982-1-23    1300.00              10

(篇幅原因,有省略……)

14 rows selected

这个需求的关键在于如何将ename员工名称列压扁为一行数据。如果不使用SQL解决,最直观的想法就是使用PL/SQL进行迭代遍历,获取到所有的数据行记录。

此时,我们就可以求助Oracle 11g中的函数listagg。首先我们来看一下listagg的函数描述(摘自Oracle SQL Reference)。

listagg的作用是将分组范围内的所有行特定列的记录加以合并成行。函数签名中的measure_expr为分组中每个列的表达式,而delimiter为合并分割符。如果delimiter不设置的话,就表示无分割符。

中间within group后面的order_by_clause表示的是进行合并中要遵守的排序顺序。而后面的over子句表明listagg是具有分析函数analyze funcation特性的。具体采用listagg有三个场景。

ü       当无分组的single-list情况下

如果要获取到deptno为30的所有员工横行记录。

SQL> select * from emp where deptno=30;

EMPNO ENAME     JOB        MGR HIREDATE         SAL     COMM DEPTNO

----- ---------- --------- ----- ----------- --------- --------- ------

7499 ALLEN     SALESMAN  7698 1981-2-20    1600.00   300.00    30

7521 WARD      SALESMAN  7698 1981-2-22    1250.00   500.00    30

7654 MARTIN    SALESMAN  7698 1981-9-28    1250.00  1400.00    30

7698 BLAKE     MANAGER   7839 1981-5-1     2850.00              30

7844 TURNER    SALESMAN  7698 1981-9-8     1500.00     0.00    30

7900 JAMES     CLERK     7698 1981-12-3     950.00              30

6 rows selected

--按照empno进行排序

SQL> select listagg(ename,‘ , ‘) within group (order byempno) from emp where deptno=30;

LISTAGG(ENAME,‘,‘)WITHINGROUP(

------------------------------------------------------------

ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

ü       在有分组条件下的listagg使用

如果要使用分组统计各个部门的所有员工列表。

SQL> select deptno, listagg(ename,‘ ,‘) within group (order by empno) from emp group by deptno;

DEPTNO LISTAGG(ENAME,‘,‘)WITHINGROUP(

------ -------------------------------------

10 CLARK ,KING ,MILLER

20 SMITH ,JONES ,SCOTT ,ADAMS ,FORD

30 ALLEN ,WARD ,MARTIN ,BLAKE ,TURNER ,JAMES

ü       使用over分组情况

如果要统计所有工作十年以上员工和他们相同部门的员工信息,就需要在listagg的基础上加入over分析函数子句。

SQL> select deptno, ename, listagg(ename, ‘ , ‘) within group (order by empno)

2 over (partition by deptno) as emp_list

3 from emp

4 where hiredate<=add_months(sysdate,-10*12);

DEPTNO ENAME     EMP_LIST

------ ------

10 CLARK     CLARK , KING , MILLER

10 KING      CLARK , KING , MILLER

10 MILLER    CLARK , KING , MILLER

20 SMITH     SMITH , JONES , SCOTT , ADAMS , FORD

20 JONES     SMITH , JONES , SCOTT , ADAMS , FORD

20 SCOTT     SMITH , JONES , SCOTT , ADAMS , FORD

20 ADAMS     SMITH , JONES , SCOTT , ADAMS , FORD

20 FORD      SMITH , JONES , SCOTT , ADAMS , FORD

30 ALLEN     ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

30 WARD      ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

30 MARTIN    ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

30 BLAKE     ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

30 TURNER    ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

30 JAMES     ALLEN , WARD , MARTIN , BLAKE , TURNER , JAMES

14 rows selected

2、lag函数“取到上个月的销售额”

我们在进行销售数据统计汇总时候,经常遇到这样的需求:“对比上月(上季度同月份或者上年度同月份),我们的销售变化情况如何?”。我们的销售数据通常是对应单月信息,如下所示。

SQL> select * from sales_qual;

MONT        QUALITIES PRICE

---------- ----------- ------

2011-01          1000 23.40

2011-02          1020 23.40

2011-03          1030 33.40

2011-04          1035 10.30

如果要获取到之前月份的信息,没有SQL专门函数就意味着需要使用PL/SQL代码进行反复的迭代获取。现在,我们可以使用lag函数来轻易实现这个功能。

lag函数是一个典型的分析函数。它提供了在不使用自连接的情况下,访问多个数据行的能力。在返回多个结果行的时候,lag函数可以访问到向上特定offset偏移行的数据。

value_expr就是访问到向上数据行进行的操作。offset是返回偏移的函数,默认值为1。over中,可以定义内部分析的顺序列。

如果我们要获取到对应上个月的销售数据,SQL语句如下:

SQL> select mont,qualities, lag(qualities,1) over (order by mont) as "Next Month Qual"

2 from sales_qual

3 order by mont;

MONT        QUALITIES Next Month Qual

---------- ----------- ---------------

2011-01          1000

2011-02          1020           1000

2011-03          1030           1020

2011-04          1035           1030

之后对销量变化率的处理就方便了,可以进行增长率比对等操作。那么,如果是上一年度或者上一季度的数据呢?我们只需要调节offset,从1变化为12或者3就可以了。

最后,对ignore/respect nulls子句的使用是什么呢?该子句的作用是确定当value_expr表达式计算出的数值为空null的时候,该列如何进行计算。ignore nulls的作用就是忽略上面计算为空的行,采用上上行row的计算结果。respect nulls的作用是直接反映为null。respect nulls为默认值。

SQL> select * from sales_qual;

MONT        QUALITIES PRICE

---------- ----------- ------

2011-04          1035 10.30

2011-05                12.30

2011-06

6 rows selected

SQL> select mont,qualities, lag(qualities,1)ignore nullsover (order by mont) as "Next Month Qual"

2 from sales_qual

3 order by mont;

MONT        QUALITIES Next Month Qual

---------- ----------- ---------------

2011-04          1035           1030

2011-05                          1035

2011-06                          1035

6 rows selected

SQL> select mont,qualities, lag(qualities,1) respect nulls over (order by mont) as "Next Month Qual"

2 from sales_qual

3 order by mont;

MONT        QUALITIES Next Month Qual

---------- ----------- ---------------

2011-04          1035           1030

2011-05                          1035

2011-06

6 rows selected

3、lead函数获取下一个月销售量

有lag的获取上个offset处理行的函数,就有lead函数处理下一个处理行的函数。lead函数实际上就是lag的逆向过程。

相关各项参数与lag函数的相同。区别就在于lead函数获取的是排序后结果集合的后offset数据行记录。

SQL> select mont,qualities, lead(qualities,1) over (order by mont) as "Next Month Qual"

2 from sales_qual

3 order by mont;

MONT        QUALITIES Next Month Qual

---------- ----------- ---------------

2011-01          1000           1020

2011-02          1020           1030

2011-03          1030           1035

2011-04          1035

4、结论

Oracle预定义函数,特别是分析函数中,有很多是非常优秀使用而且功能强大的。熟悉这个函数,在特定的环境场景下使用这些函数,可以帮我们迅速的解决很头疼的一些SQL场景。而且Oracle预定义函数的效率上也是令人满意的。

时间: 2024-10-06 00:28:13

oracle listagg函数、lag函数、lead函数 实例的相关文章

mysql的lag和Lead函数

1.mysql中没有lag和lead函数,因为Oracle使用惯了,没有这函数有些不方便.于是琢磨能不能实现. 2.如果用函数,肯定可以实现.但如果不用函数,是否能够实现呢?经过试验,发现可以达到目的. 对于lag函数,可以这么写 select @lagfield,@lagfiled:=targField,t.* from table t ,(select @lagfield:='') r 3.其中targField是你想要在下一行呈现的字段,根据字段是字符还是数字,最后的select 应该不同

oracle lag与lead分析函数简介

lag与lead函数是跟偏移量相关的两个分析函数,通过这两个函数我们可以取到当前行列的偏移N行列的值 lag可以看着是正的向上的偏移 lead可以认为负的向下的偏移 具体我们来看几个例子: 我们先看下scott的emp表的两列数据: select deptno, sal from scott.emp order by deptno DEPTNO SAL 10 2450.00 10 5000.00 10 1300.00 20 2975.00 20 3000.00 20 1100.00 20 800

[MSSQL2012]LEAD函数

LEAD函数简单点说,就是把下一行的某列数据提取到当前行来显示,看示例更能解释清楚,先看测试用脚本 DECLARE @TestData TABLE(     ID INT IDENTITY(1,1),     Department VARCHAR(20),     LastName VARCHAR(20),     Rate FLOAT ) INSERT INTO @TestData(Department,LastName,Rate) SELECT 'Document Control','Ari

PE文件结构与函数导出表——详解与实例

PE文件结构与函数导出表--详解与实例 随着windows系统从Xp升级到Win7.Win8, 从32位升级到64位,PE文件结构在整体未变的情况下发生了一些小的变动,一方面是推荐的程序装载地址未采用,另一方面,导出函数序号不再是简单的升序,而是一定程度上的进行了乱序.本文首先对PE文件结构进行了详尽的解说,接着介绍了如何得出函数导出表,整个过程采用SysWoW64目录下的wininet.dll实例进行说明.在介绍过程中,明确指出了Win7.Win8等新系统相对Xp带来的区别. 文章链接:htt

ORACLE函数之单行数字函数

 1.           ABS(X) 返回X的绝对值 SQL>SELECT ABS(-1) A,ABS(1) B,ABS(0) C FROM DUAL; A          B          C -------------------- ---------- 1          1          0 2.          ACOS(X) 返回X的反余弦值 SQL>SELECT ACOS(0) A,ACOS(0.5) B,ACOS(1) C FROM DUAL; A    

窥探Swift之函数与闭包的应用实例

今天的博客算是比较基础的,还是那句话,基础这东西在什么时候都是最重要的.说到函数,只要是写过程序就肯定知道函数是怎么回事,今天就来讨论一下Swift中的函数的特性以及Swift中的闭包.今天的一些小实例中回类比一下Objective-C中的函数的写法等等.Swift中的函数还是有许多好用的特性的,比如输入参数,使用元组返回多个值, 定义形参名,设定默认参数以及可变参数等等一些好用的特性.而在Swift中的闭包就是Objective-C中的Block, 除了语法不通外,两者的用法是一样的.废话少说

PHP5.4 文件操作函数 之 图片计数器的实例

1.访问文件或者文件夹相关函数 <?php echo filetype('c:\windows').'<br>'; ?> <?php echo filetype('c:\windows\regedit.exe').'<br>'; ?> <?php $arr=scandir("c:\\"); var_dump($arr); ?> <?php $arr=scandir("c:\\"); foreach($

问题:oracle floor;结果:Oracle的取整和四舍五入函数——floor,round,ceil,trunc使用说明

Oracle的取整和四舍五入函数——floor,round,ceil,trunc使用说明 (2011-04-06 16:10:35) 转载▼ 标签: 谈 分类: 渐行渐远 FLOOR——对给定的数字取整数位 SQL> select floor(2345.67) from dual; FLOOR(2345.67) -------------- 2345 CEIL-- 返回大于或等于给出数字的最小整数 SQL> select ceil(3.1415927) from dual; CEIL(3.14

Oracle实践--PL/SQL基础之函数

PL/SQL基础之函数 /* 函数:可以有返回值得命名的PL/SQL子程序,必须有返回值 关键字:function return */ --函数1 create or replace function helloworld return varchar2--指定返回类型,不能给定长度 as v_hello varchar2(50); begin v_hello :='helloworld!'; return v_hello;--不可少的return end; --函数调用方式: select h