ORACLE中OVER函数的用法_第1页
ORACLE中OVER函数的用法_第2页
ORACLE中OVER函数的用法_第3页
ORACLE中OVER函数的用法_第4页
ORACLE中OVER函数的用法_第5页
已阅读5页,还剩11页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

oracle over 函数详解函数详解 今天在 javaeye 上看到一道面试题 很多人都用 over 函数解决的 特意查了一下它的用法 SQL select deptno ename sal 2 from emp 3 order by deptno DEPTNO ENAME SAL 10 CLARK 2450 KING 5000 MILLER 1300 20 SMITH 800 ADAMS 1100 FORD 3000 SCOTT 3000 JONES 2975 30 ALLEN 1600 BLAKE 2850 MARTIN 1250 JAMES 950 TURNER 1500 WARD 1250 已选择 14 行 2 先来一个简单的 注意 over 条件的不同 使用 sum sal over order by ename 查询员工的薪水 连续 求和 注意 over order by ename 如果没有 order by 子句 求和就不是 连续 的 放在一起 体会一下不同之处 SQL select deptno ename sal 2 sum sal over order by ename 连续求和 3 sum sal over 总和 此处 sum sal over 等同于 sum sal 4 100 round sal sum sal over 4 份额 5 from emp 6 DEPTNO ENAME SAL 连续求和 总和 份额 20 ADAMS 1100 1100 29025 3 79 30 ALLEN 1600 2700 29025 5 51 30 BLAKE 2850 5550 29025 9 82 10 CLARK 2450 8000 29025 8 44 20 FORD 3000 11000 29025 10 34 30 JAMES 950 11950 29025 3 27 20 JONES 2975 14925 29025 10 25 10 KING 5000 19925 29025 17 23 30 MARTIN 1250 21175 29025 4 31 10 MILLER 1300 22475 29025 4 48 20 SCOTT 3000 25475 29025 10 34 20 SMITH 800 26275 29025 2 76 30 TURNER 1500 27775 29025 5 17 30 WARD 1250 29025 29025 4 31 已选择 14 行 3 使用子分区查出各部门薪水连续的总和 注意按部门分区 注意 over 条件的不同 sum sal over partition by deptno order by ename 按部门 连续 求总和 sum sal over partition by deptno 按部门求总和 sum sal over order by deptno ename 不按部门 连续 求总和 sum sal over 不按部门 求所有员工总和 效果等同于 sum sal SQL select deptno ename sal 2 sum sal over partition by deptno order by ename 部门连续求和 各部门的 薪水 连续 求和 3 sum sal over partition by deptno 部门总和 部门统计的总和 同一部门总 和不变 4 100 round sal sum sal over partition by deptno 4 部门份额 5 sum sal over order by deptno ename 连续求和 所有部门的薪水 连续 求和 6 sum sal over 总和 此处 sum sal over 等同于 sum sal 所有员工的 薪水总和 7 100 round sal sum sal over 4 总份额 8 from emp 9 DEPTNO ENAME SAL 部门连续求和 部门总和 部门份额 连续求和 总和 总份额 10 CLARK 2450 2450 8750 28 2450 29025 8 44 KING 5000 7450 8750 57 14 7450 29025 17 23 MILLER 1300 8750 8750 14 86 8750 29025 4 48 20 ADAMS 1100 1100 10875 10 11 9850 29025 3 79 FORD 3000 4100 10875 27 59 12850 29025 10 34 JONES 2975 7075 10875 27 36 15825 29025 10 25 SCOTT 3000 10075 10875 27 59 18825 29025 10 34 SMITH 800 10875 10875 7 36 19625 29025 2 76 30 ALLEN 1600 1600 9400 17 02 21225 29025 5 51 BLAKE 2850 4450 9400 30 32 24075 29025 9 82 JAMES 950 5400 9400 10 11 25025 29025 3 27 MARTIN 1250 6650 9400 13 3 26275 29025 4 31 TURNER 1500 8150 9400 15 96 27775 29025 5 17 WARD 1250 9400 9400 13 3 29025 29025 4 31 已选择 14 行 4 来一个综合的例子 求和规则有按部门分区的 有不分区的例子 SQL select deptno ename sal sum sal over partition by deptno order by sal dept sum 2 sum sal over order by deptno sal sum 3 from emp DEPTNO ENAME SAL DEPT SUM SUM 10 MILLER 1300 1300 1300 CLARK 2450 3750 3750 KING 5000 8750 8750 20 SMITH 800 800 9550 ADAMS 1100 1900 10650 JONES 2975 4875 13625 SCOTT 3000 10875 19625 FORD 3000 10875 19625 30 JAMES 950 950 20575 WARD 1250 3450 23075 MARTIN 1250 3450 23075 TURNER 1500 4950 24575 ALLEN 1600 6550 26175 BLAKE 2850 9400 29025 已选择 14 行 5 来一个逆序的 即部门从大到小排列 部门里各员工的薪水从高到低排列 累计和的规则不 变 SQL select deptno ename sal 2 sum sal over partition by deptno order by deptno desc sal desc dept sum 3 sum sal over order by deptno desc sal desc sum 4 from emp DEPTNO ENAME SAL DEPT SUM SUM 30 BLAKE 2850 2850 2850 ALLEN 1600 4450 4450 TURNER 1500 5950 5950 WARD 1250 8450 8450 MARTIN 1250 8450 8450 JAMES 950 9400 9400 20 SCOTT 3000 6000 15400 FORD 3000 6000 15400 JONES 2975 8975 18375 ADAMS 1100 10075 19475 SMITH 800 10875 20275 10 KING 5000 5000 25275 CLARK 2450 7450 27725 MILLER 1300 8750 29025 已选择 14 行 6 体会 在 from emp 后面不要加 order by 子句 使用的分析函数的 partition by deptno order by sal 里已经有排序的语句了 如果再在句尾添加排序子句 一致倒罢了 不一致 结果就令人费劲 了 如 SQL select deptno ename sal sum sal over partition by deptno order by sal dept sum 2 sum sal over order by deptno sal sum 3 from emp 4 order by deptno desc DEPTNO ENAME SAL DEPT SUM SUM 30 JAMES 950 950 20575 WARD 1250 3450 23075 MARTIN 1250 3450 23075 TURNER 1500 4950 24575 ALLEN 1600 6550 26175 BLAKE 2850 9400 29025 20 SMITH 800 800 9550 ADAMS 1100 1900 10650 JONES 2975 4875 13625 SCOTT 3000 10875 19625 FORD 3000 10875 19625 10 MILLER 1300 1300 1300 CLARK 2450 3750 3750 KING 5000 8750 8750 已选择 14 行 row number over partition by col1 order by col2 as 别 名 表示根据 col1 分组 在分组内部根据 col2 排序 而这个 别名 的值就表示每组内部排序后的顺序编号 组内连续的 唯一的 partition by col1 可省略 以 Scott tiger 登陆 以 emp 表为例 1 select deptno ename sal sum sal over order by ename 累计 按姓名排序 并将薪水逐个 累加 sum sal over 总和 此处 sum sal over 等同于 sum sal 求薪水总和 100 round sal sum sal over 4 份额 求每个人的薪水占总 额的比例 小数点后保留 2 位 括号和百分号为特殊符号 所以需要 from emp 结果如下 2 select deptno ename sal sum sal over partition by deptno order by ename 部门连续求和 partition by deptno 先按部门分组 再按姓名排序 并将薪水逐个累加 sum sal over partition by deptno 部门总和 每个部门的薪水总和 100 round sal sum sal over partition by deptno 4 部门份额 每个员工在各自部门的薪水比例 sum sal over order by deptno ename 连续求和 所有部门的薪水 连续 求和 sum sal over 总和 此处 sum sal over 等同于 sum sal 所 有员工的薪水总和 100 round sal sum sal over 4 总份额 求每个人的薪水占总额 的比例 from emp 结果如下 3 select deptno ename sal sum sal over partition by deptno order by sal dept sum 根据部门分组 再按部门内的个人薪水排序 逐个累加 sum sal over order by deptno sal sum 按部门排序 将薪水逐个累加 from emp 结果如下 4 部门从大到小排列 部门里各员工的薪水从高到低排列 select deptno ename sal sum sal over partition by deptno order by deptno desc sal desc dept sum 按部门分组后 按部门和薪水降序排 sum sal over order by deptno desc sal desc sum 按部门和薪水降 序排 from emp 结果如下 5 将各部门的员工按薪水排序 select ename job deptno sal row number over partition by deptno order by sal desc as 排名 先按部门分组 再在部门中按薪水降序排名 from scott emp 结果如下 6 查找各部门中薪水最高的前 2 位 select ename job deptno sal 排名 from select ename job deptno sal row number over partition by deptno order by sal desc as 排名 from scott emp 先将各部门的员工按薪水排序 再在结 果中取出需要的部分 where 排名 SELECT country max substr city 2 city 2 FROM 3 SELECT country sys connect by path city city 4 FROM 5 SELECT country city country rn rchild country rn 1 rfather 6 FROM 7 SELECT test country test city row number over PARTITION BY test country ORDER BY test city rn 8 FROM test 9 CONNECT BY PRIOR rchild rfather START WITH rfather LIKE 0 10 GROUP BY country COUNTRY CITY 中国 上海 台北 香港 日本 大阪 东京 怎么合并多行记录的字符串 我同事曾经建议我用 FUNCTION 来实现 但是我的报 表中有多个类似的字段需要合并 因此我想用最简单的方法来操作 没接受此方案 今天在网 上看到这样一篇文章 实在是太高兴了 而且还使用到了 partition 和聚集 这样更能丰富合 并多行记录的处理方法 下面就合并多行字符串贴出刚刚在网上搜到的一个好贴 什么是合并多行字符串 连接字符串 呢 例如 SQL desc test Name Type Nullable Default Comments COUNTRY VARCHAR2 20 Y CITY VARCHAR2 20 Y SQL select from test COUNTRY CITY 中国 台北 中国 香港 中国 上海 日本 东京 日本 大阪 要求得到如下结果集 中国 台北 香港 上海 日本 东京 大阪 实际就是对字符实现一个聚合功能 我很奇怪为什么 Oracle 没有提供官方的聚合函数来 实现它呢 下面就对几种经常提及的解决方案进行分析 有一个评测标准最高 1 被集合字段范围小且固定型 灵活性 性能 难度 这种方法的原理在于你已经知道 CITY 字段的值有几种 且还不算太多 如果太多这个 S QL 就会相当的 长 看例子 SQL select t country 2 MAX decode t city 台北 t city NULL 3 MAX decode t city 香港 t city NULL 4 MAX decode t city 上海 t city NULL 5 MAX decode t city 东京 t city NULL 6 MAX decode t city 大阪 t city NULL 7 from test t GROUP BY t country 8 COUNTRY MAX DECODE T CITY 台北 T CIT 中国 台北 香港 上海 日本 东京 大阪 大家一看 估计就明白了 如果不明白 好好补习 MAX DECODE 和分组 这种方法 无愧为最笨的方法 但是对某些应用来说 最有效的方法也许就是它 2 固定表固定字段函数法 灵活性 性能 难度 此法必须预先知道是哪个表 也就是说一个表就得写一个函数 不过方法 1 的一个取值 就要便捷多了 在大多数应用中 也不会存在大量这种合并字符串的需求 废话完毕 看 下面 定义一个函数 create or replace function str list str in in varchar2 分类字段 return varchar2 is str list varchar2 4000 default null 连接后字符串 str varchar2 20 default null 连接符号 begin for x in select TEST CITY from TEST where TEST COUNTRY str in lo op str list str list str to char x city str end loop return str list end 使用 SQL select DISTINCT T country list func1 t country from test t COUNTRY LIST FUNC1 T COUNTRY 中国 台北 香港 上海 日本 东京 大阪 SQL select t country str list t country from test t GROUP BY t country COUNTRY STR LIST T COUNTRY 中国 台北 香港 上海 日本 东京 大阪 这个时候 使用分组和求唯一都可以满足要求 它的原理就是 根据唯一的分组字段 cou ntry 在函数里面再次查询该字段对应的所有被合并列 使用 PL SQL 将其合并输出 3 灵活表函数法 灵活性 性能 难度 该方法是在方法 2 的基础上 使用动态 SQL 将表名和字段名称传入 从而达到灵活的 目的 create or replace function str list2 key name in varchar2 key in varchar2 coname in varchar2 tname in varchar2 return varchar2 as type rc is ref cursor str varchar2 4000 sep varchar2 2 val varchar2 4000 cur rc begin open cur for select coname from tname where key name x using key loop fetch cur into val exit when cur notfound str str sep val sep end loop close cur return str end SQL select test country 2 str list2 COUNTRY test country CITY TEST emplist 3 from test 4 group by test country 5 COUNTRY EMPLIST 中国 台北 香港 上海 日本 东京 大阪 4 一条 SQL 法 灵活性 性能 难度 一条 SQL 的法则是某位大师提出的 大家曾经在某个时期都乐此不彼的寻求各种的问题 一条 SQL 法 但是大师的意思似乎被曲解 很多性能差 可读性差 灵活差的 SQL 都是 这个原则产物 所谓画虎不成反成犬类 不过 解决问题始终是第一原则 这里还是给出 一个比较有代表性的一条 SQL 方法 SELECT country max substr city 2 city FROM SELECT country sys connect by path city city FROM SELECT country city country rn rchild country rn 1 rfather FROM SELECT test country test city row number over PARTITION BY test cou ntry ORDER BY test city rn FROM test CONNECT BY PRIOR rchild rfather START WITH rfather LIKE 0 GROUP BY country 下面分步解析 有 4 个 FROM 就有 4 次结果集的操作 step 1 给记录加上序号 rn SQL SELECT test country test city row number over PARTITION BY tes t country ORDER BY test city rn 2 FROM test 3 COUNTRY CITY RN 日本 大阪 1 日本 东京 2 中国 上海 1 中国 台北 2 中国 香港 3 step 2 创造子节点父节点 SQL SELECT country city country rn rchild country rn 1 rfather 2 FROM 3 SELECT test country test city row number over PARTITION BY test country ORDER BY test city rn 4 FROM test 5 日本 大阪 日本 1 日本 0 日本 东京 日本 2 日本 1 中国 上海 中国 1 中国 0 中国 台北 中国 2 中国 1 中国 香港 中国 3 中国 2 step 3 利用 sys connect by path 生成结果集 SELECT country sys connect by path city city FROM SELECT country city country rn rchild country rn 1 rfather FROM SELECT test country test city row number over PARTITION BY test cou ntry ORDER BY test city rn FROM test CONNECT BY PRIOR rchild rfather START WITH rfather LIKE 0 日本 大阪 日本 大阪 东京 中国 上海 中国 上海 台北 中国 上海 台北 香港 step 4 最终步骤 筛选结果集合 SQL SELECT country max substr city 2 city 2 FROM 3 SELECT country sys connect by path city city 4 FROM 5 SELECT country city country rn rchild country rn 1 rfather 6 FROM 7 SELECT test country test city row number

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

最新文档

评论

0/150

提交评论