窗口函数需先写PARTITION BY再ORDER BY且ORDER BY在多数数据库中不可省略执行阶段在GROUP BY后、HAVING前不能引用未SELECT或GROUP BY的列MySQL 5.7不支持须升级至8.0。窗口函数怎么写才不报错PARTITION BY 和 ORDER BY 的位置陷阱很多人一写 ROW_NUMBER() 或 SUM() OVER() 就遇到 Window function is missing ORDER BY 或 GROUP BY clause is required根本原因不是语法写错而是没理解窗口函数的执行阶段——它在 GROUP BY 之后、HAVING 之前运行但又不能依赖未出现在 SELECT 或 GROUP BY 中的列做排序。PARTITION BY 可以省略但一旦写了 ORDER BY绝大多数数据库PostgreSQL、SQL Server、MySQL 8.0就强制要求它必须存在SQLite 是个例外允许只 ORDER BY 不 PARTITION BY如果目标是“每组内按时间倒序编号”别写成 ORDER BY created_at DESC 后直接套 WHERE rn 1——这会过滤掉窗口计算前的原始行得用子查询或 CTE 包一层MySQL 5.7 不支持窗口函数强行用会报 FUNCTION xxx does not exist升级前先查 SELECT VERSION()轻量报表常用模式累计求和、同比环比、Top N 每组取一报表里最常卡壳的不是逻辑是“既要分组又要跨行计算”时不知道该用哪个函数。比如日销售额累计用 SUM(sales) OVER (ORDER BY date ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW) 最稳但若想算“比上周同期涨了多少”就得先用 LAG(sales, 7) OVER (ORDER BY date) 拿上周值再手动相减。LAG() 和 LEAD() 的第二个参数是偏移量默认为 1第三个参数是空值替代值不填就是 NULL做除法前务必 COALESCE(..., 0) 防崩要取每个品类销量 Top 3别用 GROUP BY category HAVING COUNT(*) ——那是分组计数不是排序取前几正确姿势是 codeROW_NUMBER() OVER (PARTITION BY category ORDER BY sales DESC) 外层 WHERE rn Oracle 和 PostgreSQL 支持 PERCENT_RANK()MySQL 目前还不行想兼容多库就别依赖这个函数性能雷区ORDER BY 走不走索引窗口函数会不会拖垮查询窗口函数本身不产生临时表但它的 ORDER BY 子句会触发排序操作。如果 OVER 里的排序字段没索引单次查询可能从毫秒变秒级——尤其当基础数据超百万行时。 Murf AI AI文本转语音生成工具