通过group by产生统计报告

Oracle数据库中的ROLLUP配合group by命令使用,可以提供信息汇总功能(与"小计"相似)
CUBE,也是GROUP BY子句的一种扩展,可以返回每一个列组合的小计记录,同时在末尾加上总计记录。

group by有关知识点:

1)group by rollup
2)group by cube

3)grouping和grouping_id函数

4)grouping set

示例如下:
SQL> select job,deptno,sum(sal) total_sal from emp group by (job,deptno);

JOB               DEPTNO  TOTAL_SAL
--------- ---------- ----------
MANAGER               20           2975
PRESIDENT             10           5000
CLERK                 10           1300
SALESMAN              30           5600
ANALYST               20           6000
MANAGER               30           2850
MANAGER               10           2450
CLERK                 30            950
CLERK                 20           1900

已选择9行。

18.1 rollup的用法

SQL> select job,deptno,sum(sal) total_sal from emp group by rollup(job,deptno);

JOB               DEPTNO  TOTAL_SAL
--------- ---------- ----------
CLERK                 10           1300
CLERK                 20           1900
CLERK                 30            950
CLERK                          4150
ANALYST               20           6000
ANALYST                       6000
MANAGER               10           2450
MANAGER               20           2975
MANAGER               30           2850
MANAGER                        8275
SALESMAN              30           5600
SALESMAN                       5600
PRESIDENT             10           5000
PRESIDENT                      5000
                                  29025

15 rows selected.

18.2 cube的用法
SQL> select job,deptno,sum(sal) total_sal from emp group by cube(job,deptno);

JOB               DEPTNO  TOTAL_SAL
--------- ---------- ----------
                                  29025
                      10           8750
                      20          10875
                      30           9400
CLERK                             4150
CLERK                 10           1300
CLERK                 20           1900
CLERK                 30            950
ANALYST                       6000
ANALYST               20           6000
MANAGER                        8275
MANAGER               10           2450
MANAGER               20           2975
MANAGER               30           2850
SALESMAN                       5600
SALESMAN              30           5600
PRESIDENT                      5000
PRESIDENT             10           5000

18 rows selected.

可以看出,用了rollup的group by子句所产生的所谓的超级聚合就是指在在产生聚合时会从右向左逐个对每一列进行小结,并在结果中生成独立的一行,同时也会对聚合列生成一个合计列。

这里的group by后面我们仅仅接了2列,实际上我们可以使用更多列的,这样的话oracle就会以从右向左的方式来进行逐个小结。

这里需要注意的是使用了group by和rollup后,其后面的列要用括号括起来,否则将会出现ORA-00933: SQL 命令未正确结束的错误。

看看grouping、grouping_id函数是什么?

18.3 grouping函数和grouping_id函数

GROUPING函数可以接受一列,返回0或者1。如果列值为空,那么GROUPING()返回1;如果列值非空,那么返回0。GROUPING只能在使用ROLLUP或CUBE的查询中使用。当需要在返回空值的地方显示某个值时,GROUPING()就非常有用。

GROUPING函数可以返回0,1,2,3...可以分别表示小计,合计等信息。
     
SQL>
SQL> select job,deptno,sum(sal) total_sal,grouping(job) job_grp, grouping(deptno) deptno_grp,grouping_id(job,deptno) total_grp from emp group by cube(job,deptno);

JOB               DEPTNO  TOTAL_SAL    JOB_GRP DEPTNO_GRP  TOTAL_GRP
--------- ---------- ---------- ---------- ---------- ----------
                                  29025              1          1          3
                      10           8750              1          0          2
                      20          10875              1          0          2
                      30           9400              1          0          2
CLERK                          4150                      0          1          1
CLERK                 10           1300              0          0          0
CLERK                 20           1900              0          0          0
CLERK                 30            950              0          0          0
ANALYST                        6000                      0          1          1
ANALYST               20           6000              0          0          0
MANAGER                        8275                      0          1          1
MANAGER               10           2450              0          0          0
MANAGER               20           2975              0          0          0
MANAGER               30           2850              0          0          0
SALESMAN                       5600                      0          1          1
SALESMAN              30           5600              0          0          0
PRESIDENT                      5000                      0          1          1
PRESIDENT             10           5000             0          0          0

已选择18行。

但是我们大多数情况下需要在查询的结果集的汇总列加上“合计”,怎么办呢?用grouping和grouping_id函数,然后再用decode函数判断一下是否为空就可以了

SQL>select grouping_id(job,deptno) as group_col,sum(sal) total_sal from emp group by rollup(job,deptno);

GROUP_COL  TOTAL_SAL
---------- ----------
         0       1300
         0       1900
         0        950
         1       4150
         0       6000
         1       6000
         0       2450
         0       2975
         0       2850
         1       8275
         0       5600
         1       5600
         0       5000
         1       5000
         3      29025

已选择15行。

SQL>select decode(grouping_id(job,deptno),1,'合计',3,'总计',job||deptno) as group_col,sum(sal) total_sal
from emp group by rollup(job,deptno);

GROUP_COL                                          TOTAL_SAL
------------------------------------------------- ----------
CLERK10                                                     1300
CLERK20                                                     1900
CLERK30                                                  950
合计                                                            4150
ANALYST20                                                6000
合计                                                         6000
MANAGER10                                              2450
MANAGER20                                              2975
MANAGER30                                              2850
合计                                                             8275
SALESMAN30                                             5600
合计                                                            5600
PRESIDENT10                                             5000
合计                                                         5000
总计                                                            29025

18.4 grouping sets

grouping sets用于在一个select语句中定义多个grouping, 相当于将grouping set中的多个grouping组合后再UNION ALL。

例1
SELECT null,job,mgr, AVG(sal)
FROM emp
GROUP BY(job,mgr)
union all
SELECT deptno, job, null,AVG(sal)
FROM emp
GROUP by (deptno,job);

例2=例1, 但使用了grouping sets
            
SELECT deptno,job,mgr, AVG(sal)
FROM emp
GROUP BY grouping sets ((deptno,job),(job,mgr));

DEPTNO JOB                  MGR   AVG(SAL)
---------- --------- ---------- ----------
               CLERK               7902        800
               PRESIDENT                      5000
               CLERK               7698        950
               CLERK               7788       1100
               CLERK               7782       1300
               SALESMAN            7698       1400
               MANAGER             7839     2758.33333
               ANALYST             7566       3000
        20     CLERK                           950
        30     SALESMAN                       1400
        20     MANAGER                        2975
        30     CLERK                           950
        10     PRESIDENT                      5000
        30     MANAGER                        2850
        10     CLERK                          1300
        10     MANAGER                       2450
        20     ANALYST                        3000

已选择17行。

Oracle给出了一个小表,可以表明grouping sets和 union all的关系:

GROUPING SETS Statements             Equivalent GROUP BY Statements
-------------------------------------------------------------------------------
GROUP BY GROUPING SETS(a, b, c)            GROUP BY a UNION ALL
                        GROUP BY b UNION ALL
                        GROUP BY c

GROUP BY GROUPING SETS(a, b,(b, c))            GROUP BY a UNION ALL
                        GROUP BY b UNION ALL
                        GROUP BY b, c
        
GROUP BY GROUPING SETS((a, b, c))            GROUP BY a, b, c

GROUP BY GROUPING SETS(a, (b), ())            GROUP BY a UNION ALL
                        GROUP BY b UNION ALL
                        GROUP BY ()

GROUP BY GROUPING SETS(a,ROLLUP(b, c))        GROUP BY a UNION ALL
                        GROUP BY ROLLUP(b, c)

oracle之通过group by产生统计报告的更多相关文章

  1. 在oracle中,group by后将字符拼接,以及自定义排序

    1.在oracle中,group by后将字符拼接.任务:在学生表中,有studentid和subject两个字段.要求对studentid进行group by分组,并将所选科目拼接在一起.oracl ...

  2. oracle listagg within group

    案例: 查看,每个人身上的标签. 1)表数据 2)SQL select name,listag(tag,',') within group(order by tag) tags from table_ ...

  3. Oracle 中关于 Group By 子句与多行函数嵌套搭配使用的注意事项

    目录 你需要知道的 啥叫单行函数 啥叫多行函数 如何理解这个概念 Group by 子句使用规则 看一道 071 考题 你需要知道的 提到 Group by 子句,你需要先理解一个东西:函数的分类.提 ...

  4. Oracle 表分组 group by和模糊查询like

    分组group by写法 select 字段名 from 表名 group by 字段名 查询这个字段名里的种类分组后可以加聚合函数select 字段名,聚合函数 from 表名 group by 字 ...

  5. oracle having sum group by 详解

    Aggregate functions (like SUM) often need an added GROUP BY functionality. 集合函数(类似SUM)经常需要用GROUP BY来 ...

  6. oracle select非group by的字段

    可以把group by的结果集当作一个表,然后从这里表里取数就可以了. e.g. SELECT A.PROJECT_CODE,A.DIE_NO,E.ONE_CONSUMING FROM (SELECT ...

  7. Oracle子句【group by、having】

    [分组查询]关键字:group by 分组字段名,分组字段名... --注意1:分组后,在select语句中只允许出现分组字段和多行函数 --注意2:如果是多字段分组,先按第一字段分组,然后每个小组继 ...

  8. (转)oracle group by 和order by的关系(在一起使用注意事项)

    转:http://lzfhope.blog.163.com/blog/static/636399220092554045196/ 环境:oracle 10g单单group by 或者order by本 ...

  9. 开发中mysql和oracle的区别

    首先就不描述mysql与oracle在整个数据库系统上的区别了,仅从程序员开发的角度来说: 1.主键: mysql一般会用到一个自增的属性,例如设置一个id字段,类型设置为auto increment ...

随机推荐

  1. 制作qq简易聊天框

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  2. 【AHOI 2015】 小岛 - 最短路

    描述 西伯利亚北部的寒地,坐落着由 N 个小岛组成的岛屿群,我们把这些小岛依次编号为 1 到 N . 起初,岛屿之间没有任何的航线.后来随着交通的发展,逐渐出现了一些连通两座小岛的航线.例如增加一条在 ...

  3. offer收割机也有方法论

    秋招的战火就像这夏天的温度一样炙热,陆陆续续很多学弟学妹问我秋招的注意事项,作为温暖型大叔的我此刻必须出场了. 看仔细了,接下来龙叔就把这offer收割机的秘密都告诉你们. 如果你还没点关注的话,记得 ...

  4. python header设置随机user_agent

    1 安装 fake_useragent pip install fake_useragent 2 使用 # -*- coding:utf-8 -*- from fake_useragent impor ...

  5. Linux系统添加应用服务进程的守护进程

    以前曾在Linux上维护应用服务,但是只是简单的迭代版本等工作,没有什么技术含量.最近部署在Linux服务器上的一个平台的总线进程broker(下面总线用broker指代)经常挂掉,由于总线负责服务之 ...

  6. 操作系统-I/O(6)I/O与系统调用

    所有高级语言的运行时(runtime)都提供了执行I/O功能的机制. 例如,C语言中提供了包含像printf()和scanf()等这样的标准I/O库函数, C++语言中提供了如 <<和&g ...

  7. Java类库以及它的基本组织结构

    Java 类库概念: Java 的应用程序编程接口 (API  (Application Programming Interface) )以包的形式来组织,每个包提供了大量的相关类.接口和异常处理类, ...

  8. 5分钟快速学会xpath定位

    今天我们先来和大家说一下appium,首先教大家如何定位xpath,五分钟即可学会:例:现在我想定位下面这个登录按钮: xpath该怎么写呢? 先不管三七二十几,先写//,然后找你要定位元素最近的可以 ...

  9. 力扣Leetcode 198. 打家劫舍

    打家劫舍 你是一个专业的小偷,计划偷窃沿街的房屋.每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警. 给定 ...

  10. 华师2019软件专硕复试机试题最后一题G:找数

    G. 找数 单点时限: 1.0 sec 内存限制: 256 MB 问题描述 输入一个整数 n( 2≤n≤10 ) ,你需要找到一些 n 位数(允许有前置 0 ,见样例),这些 n 位数均 由 0 ~ ...