`
lf008
  • 浏览: 66699 次
  • 性别: Icon_minigender_1
  • 来自: 福州
文章分类
社区版块
存档分类
最新评论

oracle——分析函数OVER ()

阅读更多
准备工作:
table:oracle用户scott下的emp表 ;

一.
Oracle 从8.1.6开始提供分析函数,分析函数用于计算基于组的某种聚合值,它和聚合函数的不同之处是

  对于每个组返回多行,而聚合函数对于每个组只返回一行。

exp:
select t.empno,t.ename,sum(t.sal) from emp t

这样会报:ORA-00937:非单组分组函数;
结论:聚合函数只能返回1行值。
select t.empno,t.ename,sum(t.sal)over() sum from emp t

返回的结果:
empnoenamesum
7369SMITH142781.99
7499ALLEN142781.99
7521WARD142781.99
7566JONES142781.99
7654MARTIN142781.99
7698BLAKE142781.99
7782CLARK142781.99
7788SCOTT142781.99
7839KING142781.99
7844TURNER142781.99
7876ADAMS142781.99
7900JAMES142781.99
7902FORD142781.99
7934MILLER142781.99

结论:这就是每个组返回多行。

二.
over()、over(order by...)与over(partition by...)之间的区别
1.分析函数over()用在聚合函数(max(),sun()....)后面,可返回多行所有结果的聚合值;
exp:如上;

2.over(order by...)用在聚合函数(max(),sun()....)后面,可返回根据排序结果进行统计到当前行的聚合值(即“连续”统计);
exp:
select t.empno,
       t.deptno,
       t.ename,
       t.sal,
       sum(t.sal) over(order by t.ename) sum
  from emp t

返回的结果:
empno deptno ename     sal       sum
787620ADAMS 4646.11 4646.11
749930ALLEN 8493.66 13139.77
769830BLAKE 15047.69 28187.46
778210CLARK 11823.85 40011.31
790220FORD 14935.97 54947.28
790030JAMES 4935.36 59882.64
756620JONES 15086.30 74968.94
783910KING 23841.13 98810.07
765430MARTIN 6526.80 105336.87
793410MILLER 6167.32 111504.19
778820 SCOTT 12710.16 124214.35
736920 SMITH 4089.17 128303.52
784430 TURNER 7843.77 136147.29
752130 WARD 6634.70 142781.99

分析:第二行的sum=第一行里的sal+第二行里的sal;
      第二行的sum=第一行里的sal+第二行里的sal+第三行里的sal;
      .....
结论:返回值是根据排序后的结果,当前所在的行进行统计的。

3.over(partition by...)用在聚合函数(max(),sun()....)后面,可根据pratition by里指定的某一列来统计聚合值。
exp:
select t.empno,
       t.deptno,
       t.ename,
       t.sal,
       sum(t.sal) over(partition by t.deptno) sum
  from emp t


返回的结果:
empnodeptno   ename     sal       sum
7782 10 CLARK 11823.85 41832.3
7839 10 KING 23841.13 41832.3
7934 10 MILLER 6167.32 41832.3
7369 20 SMITH 4089.17 51467.71
7876 20 ADAMS 4646.11 51467.71
7902 20 FORD 14935.97 51467.71
7788 20 SCOTT 12710.16 51467.71
7566 20 JONES 15086.30 51467.71
7499 30 ALLEN 8493.66 49481.98
7698 30 BLAKE 15047.69 49481.98
7654 30 MARTIN 6526.80 49481.98
7900 30 JAMES 4935.36 49481.98
7844 30 TURNER 7843.77 49481.98
7521 30 WARD 6634.70 49481.98

分析:每个sum的值都是把deptno相同的sal值进行求和。
结论:根据pratition by里指定的某一列来统计聚合值。

三.一个综合的例子:
exp:
question:
按部门“连续”求总和;
answer:
select t.empno,
       t.deptno,
       t.ename,
       t.sal,
       sum(t.sal) over(partition by t.deptno order by t.ename) sum
  from emp t


返回的结果:
empnodeptno   ename     sal       sum
7782 10 CLARK 11823.85 11823.85
7839 10 KING 23841.13 35664.98
7934 10 MILLER 6167.32 41832.3
7876 20 ADAMS 4646.11 4646.11
7902 20 FORD 14935.97 19582.08
7566 20 JONES 15086.30 34668.38
7788 20 SCOTT 12710.16 47378.54
7369 20 SMITH 4089.17 51467.71
7499 30 ALLEN 8493.66 8493.66
7698 30 BLAKE 15047.69 23541.35
7900 30 JAMES 4935.36 28476.71
7654 30 MARTIN 6526.80 35003.51
7844 30 TURNER 7843.77 42847.28
7521 30 WARD 6634.70 49481.98

分析:先根据partition by 进行分组,然后再根据order by 进行排序“连续”统计。

四.一个实际的例子来说明over()分析函数在代码上能简化和提高效率。
question:
         查询出管理员工人数最多的人的名字和他管理的人的名字
answer:
1.普通的方法:
select d.ename 管理员工人数最多的人的名字, o.ename 他管理的人的名字
  from emp d, emp o
 where o.empno = d.mgr
   and d.empno in
       (select p.empno
          from emp p
         where p.empno in
               (select r.mgr
                  from (select e.mgr, count(e.mgr) c from emp e group by e.mgr) r
                 where r.c is
                       (select max(w.z)
                          from (select count(m.mgr) z from emp m group by m.mgr) w)))

2.使用over()分析函数:
select e.ename 管理员工人数最多的人的名字, j.ename 他管理的人的名字
  from emp e, emp j
 where j.empno = e.mgr
   and e.empno in (select distinct (r.mgr)
                     from (select m.mgr,
                                  count(m.mgr) over(partition by m.mgr order by m.empno) t
                             from emp m) r
                    where r.t is (select max(y.h)
                               from (select count(p.mgr) over(partition by p.mgr order by p.empno) h
                                            from emp p) y))


输出结果是一样的:
管理员工人数最多的人的名字他管理的人的名字
BLAKE KING


分享到:
评论
10 楼 lf008 2010-12-04  
penggle 写道
感谢楼主分享!我在运行这些sql的时候,如果over()括号内没东西的话就报错,我用的是oracle10g的(但是在9i下确实可以的),错误代码如下:
SQL> select t.empno,t.ename,sum(t.sal)over() sum from emp t;
select t.empno,t.ename,sum(t.sal)over() sum from emp t
                                                 *
第 1 行出现错误:
ORA-00600: 内部错误代码, 参数: [12410], [], [], [], [], [], [], []


SQL>

还请各位大侠看看为什么抱着个错啊!

你好,我在10g运行了下没错,百度了下ORA-00600是oracle内核错误,你打下补丁试试
9 楼 penggle 2010-11-09  
感谢楼主分享!我在运行这些sql的时候,如果over()括号内没东西的话就报错,我用的是oracle10g的(但是在9i下确实可以的),错误代码如下:
SQL> select t.empno,t.ename,sum(t.sal)over() sum from emp t;
select t.empno,t.ename,sum(t.sal)over() sum from emp t
                                                 *
第 1 行出现错误:
ORA-00600: 内部错误代码, 参数: [12410], [], [], [], [], [], [], []


SQL>

还请各位大侠看看为什么抱着个错啊!
8 楼 lf008 2010-09-21  
hezhou_0521 写道
第一个用order by 不是更好吗?

哪个第一个
7 楼 hezhou_0521 2010-09-19  
第一个用order by 不是更好吗?
6 楼 lf008 2010-03-04  
引用
select t.empno,t.ename,sum(t.sal) from emp t group by t.empno,t.ename

这样已经报ORA-00937:非单组分组函数;
引用
select t.empno,t.ename,sum(t.sal)over()  from emp t

这样是能输出结果的,sum这一列的值是一样的都是sal的总和。
5 楼 1zebra 2010-02-20  
java_fxj 写道
select t.empno,t.ename,sum(t.sal) from emp t group by t.empno,t.ename



select t.empno,t.ename,sum(t.sal)over() sum from emp t
的结果是一样的吗?



不一样啊。
group by之后你的sum(salary) 根本就没有起到sum的效果。
4 楼 newid 2010-02-10  
当然,over()这个函数还可与很多开窗函数一起使用;如
row_number()
max()
sum()
lead()
......

各个函数的使用情况也各不一样。

有了这些函数日常工作中非常有用.
3 楼 newid 2010-02-10  
开窗函数只是在你想查询的结果中多出一列,与分组还是有很大区别的

它可以完成分组方式不能完成的一些任务。

table:id,name,money,month;一个id对应一个month一条数据
A:
select month,id , name , sum(money) over(partition by id ) from stuff ;

B:
select id , name , sum(money) from struff group id,name;

以上两种方法的结果是不一样的
A:结果是每个月都有一条数据。
B:如果要想把月查询出来就比较难了。
2 楼 風一樣的男子 2010-02-10  
不错,以前还以为用了分组函数就不能查其它字段了
1 楼 java_fxj 2010-02-09  
select t.empno,t.ename,sum(t.sal) from emp t group by t.empno,t.ename



select t.empno,t.ename,sum(t.sal)over() sum from emp t
的结果是一样的吗?

相关推荐

    ORACLE报表分析利剑——分析函数

    ### ORACLE报表分析利剑——分析函数:深度解析与应用 #### 一、理解分析函数:数据统计的新维度 分析函数作为Oracle数据库中一项高级功能,专为解决复杂报表统计需求设计,其核心优势在于能够对数据进行精细的...

    oracle语法.rar

    Oracle中可以使用ROWNUM伪列配合子查询实现分页,或者使用更现代的分析函数ROW_NUMBER(): ```sql SELECT * FROM ( SELECT t.*, ROWNUM rn FROM your_table t ORDER BY some_column ) WHERE rn BETWEEN start...

    Oracle专题培训.docx

    Oracle专题培训文档主要涵盖了Oracle数据库系统中的两个关键概念:并行执行和分析函数,以及一个相关的操作——直接路径插入。以下是对这些知识点的详细说明: 一、并行执行 并行执行是Oracle数据库提高处理大数据...

    oracle ebs开发文档

    这部分讲解了 OVER 子句在分析查询中的使用,用于计算窗口函数(如 RANK、ROW_NUMBER 等)。 **4.4 Oracle 层次树查询** 这部分介绍了如何使用 CONNECT BY 和 START WITH 子句来进行层次结构的查询,这对于组织结构...

    数据库Oracle分页显示技术

    Oracle中实现分页查询的主要方法有两种:ROWNUM和ROW_NUMBER()函数,以及Oracle的高级分页特性——FETCH NEXT和OFFSET。 1. ROWNUM方法:ROWNUM是Oracle中的一个伪列,它表示行的顺序号,从1开始。我们可以结合子...

    ORACLEEBS_FORM

    - **Over 分析查询**:介绍了如何使用OVER子句进行窗口函数计算。 - **Oracle 层次树查询**:一种特殊类型的查询,用于处理层级结构的数据。 - **Merge into 应用**:MERGE INTO语句用于根据匹配条件合并数据。 ...

    PLSQLDeveloper下载

    集成调试器(要求Oracle 7.3.4或更高)——该调试器提供您所需要的全部特性:跳入(Step In)、跳过(Step Over)、跳出(Step Out)、异常时停止运行、断点、观察和设置变量、观察全部堆栈等。基本能够调试任何...

    SQL优化实例讲解

    `Explain Plan`是Oracle数据库中用于分析查询执行计划的工具,它可以显示数据库如何执行特定的SQL语句。通过`Explain Plan`,DBA和开发人员可以检查查询的执行路径,包括是否使用了索引、扫描方式等,这对于理解查询...

    jpql-package-1.0.1

    此外,可能还增加了`OVER()`窗口函数,使得在分组、排序后的数据上进行复杂计算成为可能,如行号、排名或者移动平均等统计分析。 其次,`jpql-package-1.0.1`对相关支持的jar包进行了升级。这通常意味着内部的实现...

Global site tag (gtag.js) - Google Analytics