oracle的分析函数over 及开窗函数

转自:http://zonghl8006.blog.163.com/blog/static/4528311520083995931317/
一:分析函数over
Oracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是
对于每个组返回多行,而聚合函数对于每个组只返回一行。
下面通过几个例子来说明其应用。                                      
1:统计某商店的营业额。

date       sale
     1           20
     2           15
     3           14
     4           18
     5           30
    规则:按天统计:每天都统计前面几天的总额
    得到的结果:
DATE SALE SUM
    ----- -------- ------
    1      20        20           --1天          
    2      15        35           --1天+2天          
    3      14        49           --1天+2天+3天          
    4      18        67            .         
    5      30        97            .
    
2:统计各班成绩第一名的同学信息
    NAME   CLASS S                        
    ----- ----- ----------------------
    fda    1      80                    
    ffd    1      78                    
    dss    1      95                    
    cfe    2      74

gds    2      92                    
    gf     3      99                    
    ddd    3      99                    
    adf    3      45                    
    asdf   3      55                    
    3dd    3      78             
  
    通过:  
    --
    select * from                                                                      
    (                                                                           
    select name,class,s,rank()over(partition by class order by s desc) mm from t2
    )                                                                           
    where mm=1
    --
    得到结果:
    NAME   CLASS S                       MM                                                                                       
    ----- ----- ---------------------- ----------------------
    dss    1      95                      1                     
    gds    2      92                      1                     
    gf     3      99                      1                     
    ddd    3      99                      1

注意:
    1.在求第一名成绩的时候,不能用row_number(),因为如果同班有两个并列第一,row_number()只返回一个结果        
    2.rank()和dense_rank()的区别是:
      --rank()是跳跃排序,有两个第二名时接下来就是第四名
      --dense_rank()l是连续排序,有两个第二名时仍然跟着第三名

3.分类统计 (并显示信息)
    A   B   C                     
    -- -- ----------------------
    m   a   2                     
    n   a   3                     
    m   a   2                     
    n   b   2                     
    n   b   1                     
    x   b   3                     
    x   b   2                     
    x   b   4                     
    h   b   3
   select a,c,sum(c)over(partition by a) from t2               
   得到结果:
   A   B   C        SUM(C)OVER(PARTITIONBYA)

-- -- ------- ------------------------
   h   b   3        3                       
   m   a   2        4                       
   m   a   2        4                       
   n   a   3        6                       
   n   b   2        6                       
   n   b   1        6                       
   x   b   3        9                       
   x   b   2        9                       
   x   b   4        9                       
 
   如果用sum,group by 则只能得到
   A   SUM(C)

-- ----------------------
   h   3                     
   m   4                     
   n   6                     
   x   9                     
   无法得到B列值      
 
=====
select * from test

数据:
A B C
1 1 1
1 2 2
1 3 3
2 2 5
3 4 6

---将B栏位值相同的对应的C 栏位值加总 select a,b,c, SUM(C) OVER (PARTITION BY B) C_Sum from test

A B C C_SUM
1 1 1 1
1 2 2 7
2 2 5 7
1 3 3 3
3 4 6 6

---如果不需要已某个栏位的值分割,那就要用 null

eg: 就是将C的栏位值summary 放在每行后面

select a,b,c, SUM(C) OVER (PARTITION BY null) C_Sum from test

A B C C_SUM

1 1 1 17
1 2 2 17
1 3 3 17
2 2 5 17
3 4 6 17

求个人工资占部门工资的百分比

SQL> select * from salary;

NAME DEPT SAL

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

a 10 2000
b 10 3000
c 10 5000
d 20 4000

SQL> select name,dept,sal,sal*100/sum(sal) over(partition by dept) percent from salary;

NAME DEPT SAL PERCENT

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

a 10 2000 20
b 10 3000 30
c 10 5000 50
d 20 4000 100

二:开窗函数          
      开窗函数指定了分析函数工作的数据窗口大小,这个数据窗口大小可能会随着行的变化而变化,举例如下:
1:    
   over(order by salary) 按照salary排序进行累计,order by是个默认的开窗函数
   over(partition by deptno)按照部门分区
2:
  over(order by salary range between 5 preceding and 5 following)
   每行对应的数据窗口是之前行幅度值不超过5,之后行幅度值不超过5
   例如:对于以下列

aa
     1
     2
     2
     2
     3
     4
     5
     6
     7
     9

sum(aa)over(order by aa range between 2 preceding and 2 following)
   得出的结果是
            AA                       SUM

---------------------- -------------------------------------------------------
            1                       10                                                     
            2                       14                                                     
            2                       14                                                     
            2                       14                                                     
            3                       18                                                     
            4                       18                                                     
            5                       22                                                     
            6                       18                                                               
            7                       22                                                               
            9                       9

就是说,对于aa=5的一行 ,sum为   5-1<=aa<=5+2 的和    对于aa=2来说 ,sum=1+2+2+2+3+4=14     ;    又如 对于aa=9 ,9-1<=aa<=9+2 只有9一个数,所以sum=9    ;               3:其它:      over(order by salary rows between 2 preceding and 4 following)           每行对应的数据窗口是之前2行,之后4行 4:下面三条语句等效:                over(order by salary rows between unbounded preceding and unbounded following)           每行对应的数据窗口是从第一行到最后一行,等效: over(order by salary range between unbounded preceding and unbounded following)            等效      over(partition by null)

常用的分析函数如下所列:

row_number() over(partition by ... order by ...) rank() over(partition by ... order by ...) dense_rank() over(partition by ... order by ...) count() over(partition by ... order by ...) max() over(partition by ... order by ...) min() over(partition by ... order by ...) sum() over(partition by ... order by ...) avg() over(partition by ... order by ...) first_value() over(partition by ... order by ...) last_value() over(partition by ... order by ...) lag() over(partition by ... order by ...) lead() over(partition by ... order by ...)

示例 SQL> select type,qty from test;

TYPE QTY

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

1 6
2 9

SQL> select type,qty,to_char(row_number() over(partition by type order by qty))||'/'||to_char(count(*) over(partition by type)) as cnt2 from test;

TYPE QTY CNT2

---------- ---------- ------------
3 1/2
1 6 2/2
2 5 1/3
7 2/3
2 9 3/3

SQL> select * from test;

---------- -------------------------------------------------
1 11111
2 22222
3 33333
4 44444

SQL> select t.id,mc,to_char(b.rn)||'/'||t.id)e 2 from test t, (select rownum rn from (select max(to_number(id)) mid from test) connect by rownum <=mid ))L 4 where b.rn<=to_number(t.id) order by id

ID MC TO_CHAR(B.RN)||'/'||T.ID

--------- -------------------------------------------------- ---------------------------------------------------
1 11111 1/1
2 22222 1/2
2 22222 2/2
3 33333 1/3
3 33333 2/3
3 33333 3/3
44444 1/4 44444 2/4
4 44444 3/4CNOUG4 44444 4/4

10 rows selected

*******************************************************************

关于partition by

这些都是分析函数,好像是8.0以后才有的 row_number()和rownum差不多,功能更强一点(可以在各个分组内从1开时排序) rank()是跳跃排序,有两个第二名时接下来就是第四名(同样是在各个分组内) dense_rank()l是连续排序,有两个第二名时仍然跟着第三名。相比之下row_number是没有重复值的 lag(arg1,arg2,arg3): arg1是从其他行返回的表达式 arg2是希望检索的当前行分区的偏移量。是一个正的偏移量,时一个往回检索以前的行的数目。 arg3是在arg2表示的数目超出了分组的范围时返回的值。

1. select deptno,row_number() over(partition by deptno order by sal) from emp order by deptno;

2. select deptno,rank() over (partition by deptno order by sal) from emp order by deptno;

3. select deptno,dense_rank() over(partition by deptno order by sal) from emp order by deptno;

4. select deptno,ename,sal,lag(ename,1,null) over(partition by deptno order by ename) from emp ord er by deptno;

5. select deptno,ename,sal,lag(ename,2,'example') over(partition by deptno order by ename) from em p order by deptno;

6. select deptno, sal,sum(sal) over(partition by deptno) from emp;--每行记录后都有总计值  select deptno, sum(sal) from emp group by deptno;

7. 求每个部门的平均工资以及每个人与所在部门的工资差额

select deptno,ename,sal ,

round(avg(sal) over(partition by deptno)) as dept_avg_sal,

round(sal-avg(sal) over(partition by deptno)) as dept_sal_diff

from emp;

转载于:https://www.cnblogs.com/Mr-Simple001/p/10953767.html

Oracle 开窗函数--转相关推荐

  1. ORACLE ---开窗函数

    ORACLE -开窗函数 开窗函数: (又名:分析函数,窗口函数,OLAP函数) 开窗函数: 将数据按照一定的规则分组,统一分析各组的某项情况,每行数据返回一行结果 OLTP:事务处理 OLAP:数据 ...

  2. oracle开窗函数是什么,ORACLE数据库(六)-----开窗函数

    ORACLE数据库(六)-----开窗函数 开窗函数又名分析函数.窗口函数.OLAP(数据分析)函数 聚合函数:将数据按照一定的规则分组,统一分析各组的某项情况,每个分组返回一行结果 开窗函数:将数据 ...

  3. Oracle开窗函数over()的用法

    OVER的定义 OVER用于为行定义一个窗口,它对一组值进行操作,不需要使用GROUP BY子句对数据进行分组,能够在同一行中同时返回基础行的列和聚合列. OVER的语法 OVER ( [ PARTI ...

  4. oracle开窗函数有哪些,oracle分析函数理解(开窗函数)

    作为一个数据库管理员可以不知道分析函数,但是开发dba还是需要了解的.有时候即使经常写sql的也很少用到的分析函数. 在优化sql 的过程中,经常遇到一些通过编写复杂的sql 来实现分析函数功能的sq ...

  5. sum 开窗函数 oracle,oracle开窗函数案例详解

    一.建表和数据准备 create table t_window( name varchar2(32), orderdate varchar2(32), cost varchar2(32) ) sele ...

  6. oracle分析函数-开窗函数

    oracle分析函数 分析函数是什么? 分析函数是Oracle专门用于解决复杂报表统计需求的功能强大的函数,它可以在数据中进行分组然后计算基于组的某种统计值,并且每一组的每一行都可以返回一个统计值. ...

  7. oracle 开窗子句,分析函数和开窗函数

    分析函数 分析函数是Oracle专门用于解决复杂报表统计需求的功能强大的函数,它可以在数据中进行分组然后计算基于组的某种统计值,并且每一组的每一行都可以返回一个统计值 分析函数和聚合函数的区别 普通的 ...

  8. oracle之分析函数over及开窗函数

    2019独角兽企业重金招聘Python工程师标准>>> 一:分析函数overOracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是对于 ...

  9. oracle中over 语法,Oracle语法之OVER(PARTITIONBY..)及开窗函数

    Oracle的分析函数over 及开窗函数一:分析函数overOracle从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚 Oracle的分析函数over 及开窗函数 一:分析 ...

最新文章

  1. tomcat服务器上https的SSL证书安装配置
  2. docker操作语句
  3. 【示例】Lucene查询索引库编程步骤
  4. Gauss-Newton算法代码详细解释(转载+自己注释)
  5. AXI_04 AXI_LITE_MASTER_IP设计与验证
  6. 操作系统上机作业--多线程排序
  7. RDLC系列之五 初试XAML
  8. 双纵坐标的绘图命令_Matplotlib绘图 | 快速定义图表样式的小技巧
  9. c#中Split用法总结
  10. [转]关于管理的经典故事(员工激励)
  11. WPF学习之数据绑定
  12. python源码剖析笔记
  13. vba中find用法
  14. 【第5题】利用条件运算符的嵌套来完成此题
  15. 软件测试的基础知识点
  16. Java随机26位英文字母
  17. 湖南省工业职业技术学院 计算机,湖南工业职业技术学院
  18. Graphics.DrawRectangle 方法
  19. 给Excel表格中某一列都添加指定内容
  20. easyui简单demo

热门文章

  1. utf8 和utf8mb4 的区别
  2. 唯一索引和逻辑删除冲突
  3. Git实现从本地添加项目到远程仓库
  4. Android开发笔记(一百七十七)借助FileProvider安装应用
  5. java容器类添加元素失败失败_java容器 Set
  6. java获取访问路径、域名、项目名、请求入参
  7. 新技能Get:如何利用HTTP技术提升网页的加载速度
  8. tar ,cpio打包解压. shell脚本for,while,until循环. rpm包管理,select循环菜单,函数function,yum...
  9. spring冲刺第九天
  10. 配置 Oracle 11gR2 在 CentOS6 上开机自启动