SQL 窗口函数能否完全替代子查询?
发布时间 - 2026-01-26 00:00:00 点击率:次窗口函数能替代用于“每组内计算聚合值并保留原行”的子查询,如部门薪资排名、累计薪资、部门平均薪资等;但无法替代WHERE中的标量子查询、多层相关子查询及FROM中的表子查询。
窗口函数能替代哪些子查询?
窗口函数可以替代一部分子查询,尤其是那些用于“每组内计算聚合值并保留原行”的场景。比如要查每个部门员工的薪资排名、累计薪资、部门平均薪资等,用 ROW_NUMBER()、AVG() OVER (PARTITION BY dept) 比关联子查询或自连接更简洁高效。
但不能替代所有子查询。典型无法替代的包括:
- 子查询作为
WHERE条件中的标量子查询(如WHERE salary > (SELECT AVG(salary) FROM emp)),窗口函数无法直接参与过滤逻辑 - 多层嵌套且依赖外部作用域的子查询(如相关子查询中引用外层
WHERE条件) - 返回多行多列的子查询(如
FROM (SELECT ...) t),窗口函数只能扩展当前行,不能新增/删减行
性能差异在哪?
窗口函数通常比等效的关联子查询快,因为只需一次扫描即可完成分组内计算;而子查询在无优化时可能对每行重复执行(尤其相关子查询)。
但要注意:
- 窗口函数的
ORDER BY和大范围ROWS BETWEEN会显著增加内存和排序开销 - 某些数据库(如 MySQL 8.0 前)不支持窗口函数,强行改写会导致语法错误
- 如果只是需要单个聚合值(如整个表的平均值),用子查询
(SELE比
CT AVG(x) FROM t)
AVG(x) OVER()更轻量——后者会为每一行重复输出相同值,徒增结果集体积
哪些子查询改写后反而更难懂?
不是所有能改写的都该改。以下情况建议保留子查询:
- 业务逻辑天然分步:比如先算出“近30天活跃用户ID”,再用这些 ID 查订单,拆成两个子查询比塞进一个带
LAG()和复杂FILTER的窗口表达式更清晰 - 使用了数据库特有子查询优化(如 PostgreSQL 的
LATERAL或 Oracle 的WITH子句),强行窗口化可能丢失执行计划优势 - 需要
DISTINCT ON或TOP 1 per group但排序字段与分区键不一致时,用ROW_NUMBER() OVER (PARTITION BY ... ORDER BY ...)虽可行,但可读性常不如SELECT DISTINCT ON (group_col) ... ORDER BY group_col, priority_col
实际选型的关键判断点
决定用窗口函数还是子查询,看三个事实:
- 是否必须保留原始行数?是 → 优先窗口函数
- 计算是否依赖其他行的动态范围(如移动平均、同比环比)?是 → 窗口函数几乎是唯一选择
- 过滤条件是否基于跨行聚合结果?比如“只显示高于本部门平均薪资的员工” → 可用窗口函数先算均值,再用外层
WHERE过滤,但注意这实际是两阶段处理,不是单条语句“内联”完成
窗口函数不是子查询的升级版,而是不同抽象层级的工具。混淆它们的职责边界,容易写出既慢又难维护的 SQL。
# mysql
# oracle
# 工具
# 作用域
# sql
# select
# Filter
# postgresql
# 数据库
# 再用
# 每组
# 子句
# 尤其是
# 只需
# 只显示
# 不支持
# 会为
# 能对
# 但要
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
极客网站有哪些,DoNews、36氪、爱范儿、虎嗅、雷锋网、极客公园这些互联网媒体网站有什么差异?
google浏览器怎么清理缓存_谷歌浏览器清除缓存加速详细步骤
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
标题:Vue + Vuex 项目中正确使用 JWT 进行身份认证的实践指南
EditPlus中的正则表达式 实战(2)
微信公众帐号开发教程之图文消息全攻略
Win11怎么开启自动HDR画质_Windows11显示设置HDR选项
Laravel如何使用查询构建器?(Query Builder高级用法)
如何在IIS服务器上快速部署高效网站?
MySQL查询结果复制到新表的方法(更新、插入)
简单实现Android文件上传
5种Android数据存储方式汇总
如何快速配置高效服务器建站软件?
消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工
Laravel的.env文件有什么用_Laravel环境变量配置与管理详解
网站图片在线制作软件,怎么在图片上做链接?
javascript日期怎么处理_如何格式化输出
如何利用DOS批处理实现定时关机操作详解
php中::能调用final静态方法吗_final修饰静态方法调用规则【解答】
JS中对数组元素进行增删改移的方法总结
PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】
,网页ppt怎么弄成自己的ppt?
详解jQuery中的事件
Laravel Fortify是什么,和Jetstream有什么关系
如何用PHP快速搭建高效网站?分步指南
如何在阿里云域名上完成建站全流程?
php结合redis实现高并发下的抢购、秒杀功能的实例
php做exe能调用系统命令吗_执行cmd指令实现方式【详解】
js实现点击每个li节点,都弹出其文本值及修改
Swift开发中switch语句值绑定模式
如何快速搭建个人网站并优化SEO?
canvas 画布在主流浏览器中的尺寸限制详细介绍
微信小程序 canvas开发实例及注意事项
EditPlus中的正则表达式 实战(1)
如何挑选优质建站一级代理提升网站排名?
Laravel如何使用Telescope进行调试?(安装和使用教程)
Laravel如何处理CORS跨域问题_Laravel项目CORS配置与解决方案
如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】
如何在VPS电脑上快速搭建网站?
高端云建站费用究竟需要多少预算?
如何在阿里云ECS服务器部署织梦CMS网站?
百度浏览器网页无法复制文字怎么办 百度浏览器复制修复
Laravel全局作用域是什么_Laravel Eloquent Global Scopes应用指南
如何用西部建站助手快速创建专业网站?
如何实现javascript表单验证_正则表达式有哪些实用技巧
如何用y主机助手快速搭建网站?
Laravel如何实现事件和监听器?(Event & Listener实战)
网站制作软件免费下载安装,有哪些免费下载的软件网站?
香港服务器租用费用高吗?如何避免常见误区?
Python制作简易注册登录系统


