用 ROW_NUMBER() 替代子查询最稳妥,因其按指定排序生成唯一序号并筛选序号为1的行,支持多列排序、去重及主键保留,避免重复最大值导致多行返回或优先级失控问题。用 ROW_NUMBER() 替代子查询拿最大值行最稳直接用子查询找"最大值对应整行"容易错,尤其当有重复最大值或需要关联主键时。ROW_NUMBER() 是更可控的解法,它按指定排序给每行打唯一序号,再筛序号为 1 的行,天然支持多列排序、去重逻辑和主键保留。常见错误现象:SELECT * FROM t WHERE score = (SELECT MAX(score) FROM t) 看似简洁,但只要有多人并列最高分,就会返回多行------而你真正想要的可能只是其中一条(比如最新插入的那条);更糟的是,如果要同时取 user_id 和 created_at,这个写法根本没法控制优先级。必须显式写 ORDER BY,否则 ROW_NUMBER() 行为不可预测(不同数据库默认策略不同)排序字段里建议包含主键(如 ORDER BY score DESC, id DESC),避免因相同分数导致窗口函数分配顺序不一致MySQL 8.0+、PostgreSQL、SQL Server 都支持;SQLite 3.25+ 也支持,但旧版 SQLite 只能退化用相关子查询SELECT id, name, scoreFROM ( SELECT id, name, score, ROW_NUMBER() OVER (ORDER BY score DESC, id DESC) AS rn FROM users) rankedWHERE rn = 1;为什么不能只靠 MAX() + 关联子查询MAX() 只返回标量值,它本身不携带任何行上下文。想靠它反查原表某一行,就得额外做一次 JOIN 或相关子查询,性能差、逻辑绕、还容易漏数据。使用场景:比如日志表中查"每个用户最后一条操作记录",有人会写:SELECT * FROM logs l1 WHERE l1.created_at = (SELECT MAX(l2.created_at) FROM logs l2 WHERE l2.user_id = l1.user_id) ------ 这在小表上勉强能跑,但一旦 user_id 和 created_at 缺少联合索引,执行计划大概率是嵌套循环 + 全表扫描。相关子查询对每一行都触发一次内层查询,复杂度接近 O(n2)即使加了索引,优化器也不总能正确识别"最大时间戳 + 分组"的意图,常退化为临时表或文件排序若存在毫秒级时间戳重复,仍可能返回多行,且无法控制取哪一条用 JOIN + 聚合结果反查的坑有人试图先聚合出最大值及对应主键:SELECT user_id, MAX(created_at) FROM logs GROUP BY user_id,再用这个结果 JOIN 原表。问题在于:聚合结果里没有原表的其他字段(比如 action 或 ip),JOIN 后还得处理"多个同时间戳记录怎么选"的问题。 通义听悟 阿里云通义听悟是聚焦音视频内容的工作学习AI助手,依托大模型,帮助用户记录、整理和分析音视频内容,体验用大模型做音视频笔记、整理会议记录。
相关推荐
呱呱复呱呱2 小时前
Django CBV 源码解读:一个请求是怎么找到你的 get() 方法的Nturmoils3 小时前
订单列表慢查询,先看 WHERE、ORDER BY 和 LIMIT曲幽6 小时前
刚部署的 LibreTranslate 频频翻车?我掏出了 20 年前的 StarDict 词典,用 FastAPI 搭了个本地词典翻译 API渣波7 小时前
拒绝 SQL 焦虑!手把手带你用 NestJS + Prisma + DTO 写出“防弹”级后端代码荣码7 小时前
用Streamlit给AI应用套个界面,10行代码出Web页面兵慌码乱16 小时前
基于Python+PyQt5+SQLite的药房管理系统实现:事务一致性与界面解耦全流程解析金銀銅鐵18 小时前
[Python] 体验用欧几里得算法计算最大公约数的过程FreakStudio1 天前
W55MH32L-EVB 上手测评:硬件 TCP/IP 加持的以太网单片机,MicroPython 零门槛开发用户0332126663671 天前
使用 Python 从零创建 Word 文档Csvn1 天前
Python 两大经典坑点 —— 可变默认参数 & 闭包延迟绑定