详解Oracle查询中OVER (PARTITION BY ..)用法

本文主要介绍了oracle查询中over (partition by ..)用法,内容和代码大家参考一下,希望能帮助到大家。为了方便大家学习和测试,所有的例子都是在oracle自带用户scott下建立的。

注:标题中的红色order by是说明在使用该方法的时候必须要带上order by。

一、rank()/dense_rank() over(partition by …order by …)

现在客户有这样一个需求,查询每个部门工资最高的雇员的信息,相信有一定oracle应用知识的同学都能写出下面的SQL语句:

select e.ename, e.job, e.sal, e.deptno  from scott.emp e,     (select e.deptno, max(e.sal) sal from scott.emp e group by e.deptno) me  where e.deptno = me.deptno   and e.sal = me.sal;

在满足客户需求的同时,大家应该习惯性的思考一下是否还有别的方法。这个是肯定的,就是使用本小节标题中rank() over(partition by…)或dense_rank() over(partition by…)语法,SQL分别如下:

select e.ename, e.job, e.sal, e.deptno  from (select e.ename,         e.job,         e.sal,         e.deptno,         rank() over(partition by e.deptno order by e.sal desc) rank      from scott.emp e) e  where e.rank = 1;

select e.ename, e.job, e.sal, e.deptno  from (select e.ename,         e.job,         e.sal,         e.deptno,         dense_rank() over(partition by e.deptno order by e.sal desc) rank      from scott.emp e) e  where e.rank = 1;

为什么会得出跟上面的语句一样的结果呢?这里补充讲解一下rank()/dense_rank() over(partition by e.deptno order by e.sal desc)语法。

over: 在什么条件之上。

partition by e.deptno: 按部门编号划分(分区)。

order by e.sal desc: 按工资从高到低排序(使用rank()/dense_rank() 时,必须要带order by否则非法)

rank()/dense_rank(): 分级

整个语句的意思就是:在按部门划分的基础上,按工资从高到低对雇员进行分级,“级别”由从小到大的数字表示(最小值一定为1)。

那么rank()和dense_rank()有什么区别呢?

rank(): 跳跃排序,如果有两个第一级时,接下来就是第三级。

dense_rank(): 连续排序,如果有两个第一级时,接下来仍然是第二级。

小作业:查询部门最低工资的雇员信息。

二、min()/max() over(partition by …)

现在我们已经查询得到了部门最高/最低工资,客户需求又来了,查询雇员信息的同时算出雇员工资与部门最高/最低工资的差额。这个还是比较简单,在第一节的groupby语句的基础上进行修改如下:

蓝心千询 蓝心千询

蓝心千询是vivo推出的一个多功能AI智能助手

蓝心千询 34 查看详情 蓝心千询

select e.ename,      e.job,      e.sal,      e.deptno,      e.sal - me.min_sal diff_min_sal,      me.max_sal - e.sal diff_max_sal   from scott.emp e,      (select e.deptno, min(e.sal) min_sal, max(e.sal) max_sal       from scott.emp e       group by e.deptno) me   where e.deptno = me.deptno   order by e.deptno, e.sal;

上面我们用到了min()和max(),前者求最小值,后者求最大值。如果这两个方法配合over(partition by …)使用会是什么效果呢?大家看看下面的SQL语句:

select e.ename,     e.job,     e.sal,     e.deptno,     nvl(e.sal - min(e.sal) over(partition by e.deptno), 0) diff_min_sal,     nvl(max(e.sal) over(partition by e.deptno) - e.sal, 0) diff_max_sal  from scott.emp e;

这两个语句的查询结果是一样的,大家可以看到min()和max()实际上求的还是最小值和最大值,只不过是在partition by分区基础上的。

小作业:如果在本例中加上order by,会得到什么结果呢?

三、lead()/lag() over(partition by … order by …)

中国人爱攀比,好面子,闻名世界。客户更是好这一口,在和最高/最低工资比较完之后还觉得不过瘾,这次就提出了一个比较变态的需求,计算个人工资与比自己高一位/低一位工资的差额。这个需求确实让我很是为难,在groupby语句中不知道应该怎么去实现。不过。。。。现在我们有了over(partition by …),一切看起来是那么的简单。如下:

select e.ename,     e.job,     e.sal,     e.deptno,     lead(e.sal, 1, 0) over(partition by e.deptno order by e.sal) lead_sal,     lag(e.sal, 1, 0) over(partition by e.deptno order by e.sal) lag_sal,     nvl(lead(e.sal) over(partition by e.deptno order by e.sal) - e.sal,       0) diff_lead_sal,     nvl(e.sal - lag(e.sal) over(partition by e.deptno order by e.sal), 0) diff_lag_sal  from scott.emp e;

看了上面的语句后,大家是否也会觉得虚惊一场呢(惊出一身冷汗后突然鸡冻起来,这样容易感冒)?我们还是来讲解一下上面用到的两个新方法吧。

lead(列名,n,m): 当前记录后面第n行记录的的值,没有则默认值为m;如果不带参数n,m,则查找当前记录后面第一行的记录的值,没有则默认值为null。

lag(列名,n,m): 当前记录前面第n行记录的的值,没有则默认值为m;如果不带参数n,m,则查找当前记录前面第一行的记录的值,没有则默认值为null。

下面再列举一些常用的方法在该语法中的应用(注:带order by子句的方法说明在使用该方法的时候必须要带order by):

select e.ename,     e.job,     e.sal,     e.deptno,     first_value(e.sal) over(partition by e.deptno) first_sal,     last_value(e.sal) over(partition by e.deptno) last_sal,     sum(e.sal) over(partition by e.deptno) sum_sal,     avg(e.sal) over(partition by e.deptno) avg_sal,     count(e.sal) over(partition by e.deptno) count_num,     row_number() over(partition by e.deptno order by e.sal) row_num  from scott.emp e;

大家在读完本片文章之后可能会有点误解,就是OVER (PARTITION BY ..)比GROUP BY更好,实际并非如此,前者不可能替代后者,而且在执行效率上前者也没有后者高,只是前者提供了更多的功能而已,所以希望大家在使用中要根据需求情况进行选择。

相关推荐:

Oracle程序开发小技巧

Oracle使用触发器和mysql中使用触发器的案例比较

oracle数据库常用的99条查询语句

以上就是详解Oracle查询中OVER (PARTITION BY ..)用法的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/387407.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月6日 14:16:24
下一篇 2025年11月6日 14:17:12

相关推荐

  • OAuth响应处理与安全会话管理:数据库集成与Cookie最佳实践

    本教程深入探讨了OAuth2认证流程结束后,如何安全高效地处理用户数据并建立会话。文章首先介绍了将OAuth提供者返回的用户数据存储到数据库的最佳实践,重点讲解了原子性的UPSERT操作以避免数据冗余和竞态条件。随后,详细阐述了基于Cookie的会话管理策略,强调了使用HTTPS、Secure、Ht…

    2025年12月16日
    000
  • Golang测试代码生成 自动化生成测试用例

    自动化生成Golang测试用例的核心在于结合Fuzzing、反射和代码生成工具。Go内置Fuzzing能自动探索输入并发现崩溃,解决输入多样性问题;反射可用于自动生成复杂结构体的测试数据,提升数据准备效率;gopter等PBT库则通过定义代码属性验证逻辑正确性;而外部依赖模拟和“神谕问题”仍需人工设…

    2025年12月15日
    000
  • Golang的crypto库如何实现数据加密 演示AES和RSA最佳实践

    golang的crypto库为数据加密提供了坚实的基础,它不是一个单一的“加密”功能,而是一系列密码学原语的集合。在实际应用中,aes(高级加密标准)凭借其对称加密的高效性,成为处理大量数据的首选,尤其是在gcm模式下,它能同时提供数据的机密性、完整性和认证。而rsa(rivest-shamir-a…

    2025年12月15日 好文分享
    000
  • Golang如何优化加密运算 使用硬件加速指令提升AES性能

    要优化golang中的aes加密性能,关键在于利用硬件加速和高效模式。1. 使用标准库crypto/aes包,其已自动启用aes-ni指令;2. 确认cpu支持aes-ni;3. 采用gcm模式提升性能与安全性;4. 复用cipher实例避免重复初始化;5. 利用并发处理大数据块;6. 减少内存拷贝…

    2025年12月15日 好文分享
    000
  • Debian系统如何监控JSP应用的运行状态

    在Debian操作系统中,对JSP应用进行运行状态监控可以采用多种方式和相关工具。以下是一些常见的方法: 1. 利用SpringBoot内置的性能监控功能 如果JSP项目是基于SpringBoot构建的,可以使用其自带的性能监控组件。SpringBoot提供了一些基础的监控类,比如Performan…

    2025年12月15日
    000
  • Debian下sqlplus使用技巧有哪些

    在Debian操作系统中使用SQL*Plus时,掌握一些实用技巧可以显著提升操作效率和管理体验。以下是一些常用的技巧: 增强命令行操作:通过安装rlwrap工具来实现命令历史浏览和上下文切换功能。具体步骤如下: sudo apt-get install rlwrap echo “alias sqlp…

    2025年12月15日
    000
  • 如何在Debian中提高phpstorm的稳定性

    在Debian系统中增强PhpStorm的稳定性可通过以下几种方式实现: 维持软件更新: 定期将PhpStorm升级至最新版本,例如PhpStorm 2018.1版本包含了大量错误修正及功能优化。同时也要确保Debian系统的全面更新,这样可以有效规避潜在的兼容性冲突。 选择合适的JVM: PhpS…

    2025年12月15日
    000
  • 使用Go语言连接Oracle数据库时是否需要安装Oracle客户端?

    Go语言连接Oracle数据库:是否必须安装Oracle客户端? 在Go语言开发中连接Oracle数据库是常见需求。许多教程和示例都建议安装Oracle客户端,这在Windows桌面开发和Linux生产环境中可能带来配置上的差异和不便。那么,Go语言的Oracle数据库驱动程序能否绕过Oracle客…

    2025年12月15日
    000
  • 在Go语言中使用Oracle驱动是否需要安装Oracle客户端?

    Go语言连接Oracle数据库:是否需要Oracle客户端? 许多Go语言开发者在连接Oracle数据库时,都会纠结于是否需要安装Oracle客户端。网上的许多示例都依赖于Oracle客户端,这无疑增加了跨平台部署的复杂性(例如,开发环境为Windows,生产环境为Linux)。 那么,Go语言的O…

    2025年12月15日
    000
  • 在Go中使用Oracle数据库驱动是否需要安装Oracle客户端?

    Go语言连接Oracle数据库:Oracle客户端并非必需 在Go语言开发中连接Oracle数据库,开发者常面临一个关键问题:是否必须安装Oracle客户端?尤其在Windows开发环境与Linux生产环境切换时,这个问题显得尤为重要。 许多教程都包含安装Oracle客户端的步骤,这无疑增加了开发和…

    2025年12月15日
    000
  • python oracle数据库如何安装?

    首先安装Oracle Instant Client并配置环境变量,再通过pip install oracledb安装Python驱动,最后用代码测试连接;确保客户端版本与系统匹配,并正确设置PATH、LD_LIBRARY_PATH或DYLD_LIBRARY_PATH以避免常见错误。 要在Python…

    2025年12月14日
    000
  • python-oracledb 游标对象与数据库会话管理深度解析

    本文深入探讨 `python-oracledb` 库中游标对象(Cursor Object)及其变量(Cursor Variable)的工作原理与生命周期。我们将阐明 `cursor.var()` 创建的变量在 Python 客户端和 Oracle 数据库会话之间的关系,纠正关于其值持久性的常见误解…

    2025年12月14日
    000
  • SQLAlchemy 声明式模型中指定数据库表模式(Schema)的方法

    本文详细介绍了如何在使用 sqlalchemy 声明式 api 定义和创建数据库表时,指定表所属的数据库模式(schema)。通过在声明式模型类中利用 `__table_args__` 属性并设置 `schema` 参数,开发者可以精确控制表在数据库中的位置,从而避免默认的“public”模式,尤其…

    2025年12月14日
    000
  • python-oracledb 游标与绑定变量:连接管理与数据持久化解析

    本文深入探讨了 `python-oracledb` 中游标对象 (`cursor`) 和绑定变量 (`cursor.var()`) 的工作机制及其生命周期。我们将澄清绑定变量在客户端Python环境与服务端Oracle数据库会话之间的行为差异,特别是数据在连接断开与重连后是否保持的问题。文章还将提供…

    2025年12月14日
    000
  • 深入理解 python-oracledb 中的游标对象与变量绑定

    `python-oracledb` 的 `cursor.var()` 方法用于创建客户端绑定变量。这些变量是 Python 对象,其值在客户端内存中维护,并不会因数据库连接的关闭而自动丢失。只有当变量通过游标执行 SQL 语句时,其值才与数据库会话进行交互。理解这一客户端与服务器端的区别,对于正确管…

    2025年12月14日
    000
  • python-oracledb 游标对象详解:生命周期、绑定变量与连接管理

    本文深入探讨 `python-oracledb` 中游标对象(cursor)和绑定变量(bind variables)的工作机制。我们将阐明 `cursor.var()` 如何创建客户端 Python 对象以管理绑定变量,并解释数据库会话与游标的生命周期。通过示例代码,纠正关于连接关闭与重开后变量值…

    2025年12月14日
    000
  • SQLAlchemy声明式风格下如何指定数据库表模式

    本文详细阐述了如何在sqlalchemy的声明式风格中,为数据库表指定特定的schema。通过利用模型类中的`__table_args__`属性,开发者可以设置`schema`参数,从而控制表在postgresql等支持schema的数据库中的命名空间归属。这使得表能够被创建到指定的schema而非…

    2025年12月14日
    000
  • SQLAlchemy声明式模型中指定数据库表Schema的方法

    本文详细介绍了如何在sqlalchemy的声明式模型中为数据库表指定schema。通过在模型类中利用`__table_args__`属性,开发者可以轻松地将表关联到特定的数据库schema,从而更好地组织和管理数据库结构,尤其适用于支持schema的数据库如postgresql,并提供了完整的代码示…

    2025年12月14日
    000
  • 使用SQLAlchemy声明式ORM指定数据库表Schema的教程

    本文详细介绍了如何在sqlalchemy声明式orm中为数据库表指定特定的schema,而非使用数据库的默认schema。通过利用模型类中的`__table_args__`属性,并设置`schema`参数,开发者可以灵活地控制表在不同命名空间中的创建位置。文章将提供具体的代码示例和使用指南,并探讨此…

    2025年12月14日
    000
  • Python代码如何连接MySQL数据库 Python代码使用PyMySQL驱动的连接方法

    答案:PyMySQL是纯Python实现的MySQL驱动,安装简单、跨平台兼容性好,支持参数化查询和DictCursor返回字典结果,避免SQL注入并提升代码可读性;实际项目中应通过环境变量或配置文件管理数据库凭证以确保安全,并使用DBUtils等工具构建连接池提升高并发场景下的性能;处理大数据量时…

    2025年12月14日
    000

发表回复

登录后才能评论
关注微信