Oracle中ROW_NUMBER()OVER()函数用法实例讲解

 更新时间:2023年04月25日 09:26:18   作者:遥远的救世主○  
Oracle中的row_number()函数用于在查询结果中为每一行生成一个唯一的行号,下面这篇文章主要给大家介绍了关于Oracle中ROW_NUMBER()OVER()函数用法的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下

Oracle中ROW_NUMBER() OVER()函数用法

1. 说明:

ROW_NUMBER() OVER() 函数的作用:分组排序

2. 原理:

row_number() over() 函数,over() 里的分组以及排序的执行晚于 where、group by、order by 的执行。

3.语法:

row_number() over( partition by 分组列 order by 排序列 desc )

4.示例一:

查询表:SELECT * FROM SCOTT.EMP ;

使用Row_number() over() 函数,排序

SELECT EMPNO,ENAME,SAL,DEPTNO,Row_number() over( order by sal) rs FROM
SCOTT.EMP ;

根据工资排序并添加序号

5. 示例二

1.建立测试学生数据表

create table Students 
(
id int,
name varchar2(100),
classid int,
score int
);
 
insert into Students values(1, '学生1', 1, 88);
insert into Students values(2, '学生2', 3, 68);
insert into Students values(3, '学生3', 1, 78);
insert into Students values(4, '学生4', 2, 87);
insert into Students values(5, '学生5', 1, 89);
insert into Students values(6, '学生6', 2, 91);
insert into Students values(7, '学生7', 3, 67);
insert into Students values(8, '学生8', 1, 77);
insert into Students values(9, '学生9', 3, 77);
commit;

2.查学生数据根据班级分组,再根据分数排名。获取到每个班级的学生分数排名

select id, name, classid, score, row_number() over(partition by classid order by score desc) rank from Students;

3. 获取到每个班级分数排名第一的学生

select * from (select id, name, classid, score, row_number() over(partition by classid order by score desc) rank from Students) where rank = 1;

重点说明:

  1. parttion by 是 Oracle 中分析性函数的一部分,用于给结果集进行分区,它和聚合函数 group by
    不同的地方在于它只是将原始数据进行名次排列,能够返回一个分组中的多条记录(记录数不变),而 group by
    是对原始数据进行聚合统计,一般只有一条反映统计值的结果(每组返回一条)。
  2. over() 必须有 ORDER BY 语句
  3. 分组内从 1开始排序
  4. over() 中的排序字段为空,会被排到第一

例如:

将学生1的分数设置为 null,再获取到分组班级的学生分数排名

select id, name, classid, score, row_number() over(partition by classid order by score desc) rank from Students;

修改学生一在该班级的排序,分数最低排最后即可修正这个问题

select id, name, classid, score, row_number() over(partition by classid order by score desc nulls last) rank from Students;

2. 分析函数的例子二:

2.1 分析函数的形式:

分析函数带有一个开窗函数over(),包含三个分析子句:分组(partition by),排序(order by), 窗口(rows),他们的使用形式如下:

over(partition by xxx order by yyy rows between zzz)
– 例如在scott.emp表中:xxx为deptno, yyy为sal,
– zzz为unbounded preceding and unbounded following

分析函数的例子:

显示各部门员工的工资,并附带显示该部分的最高工资。

SQL如下:

SELECT DEPTNO, EMPNO, ENAME, SAL, LAST_VALUE(SAL) OVER (PARTITION BY DEPTNO 
ORDER BY SAL ROWS BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING) MAX_SAL 
FROM EMP;

注: current row 表示当前行

    unbounded preceding  表示第一行

    unbounded following    表示最后一行

last_value(sal) 的结果与 order by sal 排序有关。如果排序为order by sal desc, 则最终的结果为分组排序后sal的最小值(分组排序后的最后一个值), 当deptno为10时,max_sal为1300

2.2 两个order by 的执行机制

分析函数是在整个SQL查询结束后(SQL语句中的order by 的执行比较特殊)再进行的操作,也就是说SQL语句中的order by也会影响分析函数的执行结果:

两者一致:如果SQL语句中的order by 满足分析函数分析时要求的排序,那么SQL语句中的排序将先执行,分析函数在分析时就不必再排序。
两者不一致:如果SQL语句中的order by 不满足分析函数分析时要求的排序,那么SQL语句中的排序将最后在分析函数分析结束后执行排序。

2.3 分析函数中的分组、排序、窗口

分析函数包含三个分析子句:分组(partition by)、排序(order by)、窗口(rows)。

窗口就是分析函数分析时要处理的数据范围,就拿sum来说,它是sum窗口中的记录而不是整个分组中的记录。因此我们在想得到某个栏位的累计值时,我们需要把窗口指定到该分组中的第一行数据到当前行,如果你指定该窗口从该分组中的第一行到最后一行,那么该组中的每一个sum值都会一样,即整个组的总和。

窗口子句中我们经常用到指定第一行,当前行,最后一行这样的三个属性:

第一行是 unbounded preceding

当前行是 current row

最后一行是 unbounded following

窗口子句不能单独出现,必须有order by 子句时才能出现,如:

LAST_VALUE(SAL) OVER (PARTITION BY DEPTNO ORDER BY SAL ROWS BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING )

以上示例指定窗口为整个分组.

而出现order by 子句的时候,不一定要有窗口子句,但效果会不一样,此时窗口默认是当前组的第一行到当前行!

SQL语句为:

SELECT DEPTNO, EMPNO, ENAME, SAL,
last_value(SAL) OVER(PARTITION BY DEPTNO ORDER BY SAL) MAX_SAL FROM EMP;

等价于:

SELECT DEPTNO, EMPNO, ENAME, SAL,last_value(SAL) OVER(PARTITION BY DEPTNO 
ORDER BY SAL ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) MAX_SAL FROM EMP;

结果如下图:

当省略窗口子句时:

  • 如果存在order by, 则默认的窗口是 unbounded preceding and current row.
  • 如果同时省略order by, 则默认的窗口是 unbounded preceding and unbounded following.

如果省略分组,则把全部记录当成一个组:

  • 如果存在order by 则默认窗口是unbounded preceding and current row
  • 如果这时省略order by 则窗口默认为 unbounded preceding and unbounded following

可参考:https://www.jb51.net/article/282335.htm

总结

到此这篇关于Oracle中ROW_NUMBER()OVER()函数用法的文章就介绍到这了,更多相关Oracle中ROW_NUMBER()OVER()函数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • oracle 多个字符替换实现

    oracle 多个字符替换实现

    CSDN上的一个网友,需要一个sql语句的解决方案需求是这样的求写oracle多个字符替换(有测试数据)
    2009-10-10
  • 分享Oracle 11G Client 客户端安装步骤(图文详解)

    分享Oracle 11G Client 客户端安装步骤(图文详解)

    这篇文章主要介绍了分享Oracle 11G Client 客户端安装步骤(图文详解),非常具有实用价值,需要的朋友可以参考下。
    2016-12-12
  • Oracle 添加用户并赋权,修改密码,解锁,删除用户的方法

    Oracle 添加用户并赋权,修改密码,解锁,删除用户的方法

    Oracle 添加用户并赋权,修改密码,解锁,删除用户实现方法,需要的朋友可以参考下。
    2009-10-10
  • Oracle CDB管理实现多租户管理功能

    Oracle CDB管理实现多租户管理功能

    这篇文章主要介绍了Oracle CDB管理实现多租户管理功能的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用oracle具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2022-12-12
  • Oracle数据库数据丢失恢复的几种方法总结

    Oracle数据库数据丢失恢复的几种方法总结

    相信大家无论是开发、测试还是运维过程中,都可能会因为误操作、连错数据库、用错用户、语句条件有误等原因,导致错误删除、错误更新等问题。当你捶胸顿足或吓得腿软时,肯定希望有办法来恢复这些数据。oracle就提供了一些强大的方法或机制,可以帮到有需要的你。
    2016-12-12
  • Oracle中update和select 关联操作

    Oracle中update和select 关联操作

    本文主要向大家介绍了Oracle数据库之oracle update set select from 关联更新,通过具体的内容向大家展现,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧
    2022-01-01
  • Oracle新建用户、角色,授权,建表空间的sql语句

    Oracle新建用户、角色,授权,建表空间的sql语句

    Oracle创建用户操作相信大家都不陌生,下面就为您介绍Oracle创建用户的语法的相关知识,希望对您学习Oracle创建用户的方面能有所帮助
    2012-07-07
  • Oracle数据库用户密码过期的解决方法

    Oracle数据库用户密码过期的解决方法

    常使用oracle数据库的小伙伴可能会经常遇到oracle密码过期的问题,下面这篇文章主要给大家介绍了关于Oracle数据库用户密码过期的解决方法,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-05-05
  • Oracle数据库执行慢问题排查详细过程

    Oracle数据库执行慢问题排查详细过程

    这篇文章主要给大家介绍了关于Oracle数据库执行慢问题排查的详细过程,在企业级应用中,数据库的稳定性和性能是至关重要的,文中给出了详细的代码实例,需要的朋友可以参考下
    2023-07-07
  • oracle指定排序的方法详解

    oracle指定排序的方法详解

    本篇文章是对oracle指定排序的方法进行了详细的分析介绍,需要的朋友参考下
    2013-05-05

最新评论