Oracle 中 ROWNUM 使用问题记录

ROWNUM 使用问题记录(2023-08-17)

  • Oracle 版本: 19.0.0.0.0 Enterprise
  • 现象:今天在项目遇到一个问题,测试人员反馈前一天能看到的数据今天看不到了

用表格举例,这是前一天看到的数据,有9、7、1 这几个数量信息

日期 ID 数量
XXX 1 9
XXX 2 7
XXX 3 1

今天测试反馈 9 没了,但是库里是有的

ID 数量
XXX 2 7
XXX 3 1
XXX 4 1
  • 排查问题:我跟踪代码把SQL拿出来到数据库执行,确实没查出来数值 9 的数据
sql 复制代码
# SQL 如下(昨天能显示的时候ROWNUM 为<=20)
SELECT dt_date,id,num FROM T 
	WHERE dt_date>=2301 and dt_date<=2307 AND ROWNUM <= 10
	ORDER BY num DESC
  • 比较奇怪的是 ROWNUM<= 20 的时候能查出来为9 的那一条数据(其实即使查出来 9 数据结果也不对,因为还有一条数值为 12 的)
    • 先是改回 ROWNUM <= 20 确实能看到数值为9 的那一条数据了,但是还有一个数值12的没有查出来,还是不对
    • 通过回忆MYSQL的写法结合查询结果现象推测Oracle SQL 执行顺序与MYSQL不一样
      • 在MYSQL中上述SQL执行结果是正常的会取出数值为9 、12 的数据,MySql 会先执行Order By 操作,然后执行 Limit 取值,并且 Limit 会在Where 条件后执行
      • 在Oracle中就不行了,这里使用到了ROWNUM,而 ROWNUM 是写在 Where 条件中的,SQL 执行过程与Mysql 就产生了差异,根据现象得知Oracle先得到结果,然后生成虚拟的ROWNUM,再然后执行WHERE条件,最后 Orader By 操作,这样就出现了与测试反馈的一样的结果,取值在前排序在后,所以取出来的值如果刚巧数据满足要求时就不容易看出来问题
  • 解决问题:通过包装一层SQL来处理这个问题,先排序,再取值
sql 复制代码
# 通过嵌套SQL查询解决
SELECT * FROM (
  SELECT dt_date,id,num FROM T 
	WHERE dt_date>=2301 and dt_date<=2307
 	ORDER BY num DESC
) A
WHERE ROWNUM<=10
  • 本次问题根因:
  1. 太久(十多年了)没有使用过Oracle 了特性啥的早忘光光了,不然应该不会犯这个错误
  2. SQL 是老师提供的,我也没想那么多,直接用了,估计老师也可能也没想到吧
相关推荐
kingwebo'sZone37 分钟前
Error parsing column 10 (YingShou=-99.5 - Double) dapper sqlite
数据库·sqlite
Austindatabases1 小时前
给阿里云MongoDB 的感谢信 !!成本降低80%
数据库·mongodb·阿里云·云计算
lyrhhhhhhhh2 小时前
JDBC工具类的三个版本
数据库
正在走向自律2 小时前
【金仓数据库征文】政府项目数据库迁移:从MySQL 5.7到KingbaseES的蜕变之路
数据库·mysql·kingbasees·金仓数据库 2025 征文·数据库平替用金仓
黑客笔记3 小时前
sql注入漏洞的对抗
数据库·sql·测试工具
tianlebest3 小时前
Laravel 安全:批量赋值 fillable 与 guarded
数据库·安全·laravel
小叶子来了啊3 小时前
软考(信息系统运行管理员)
数据库
chunfeng—3 小时前
Redis 主从同步与对象模型(四)
数据库·redis·缓存·集群·哨兵
Leo.yuan4 小时前
热力图是什么?三分钟学会热力图数据分析怎么做!
大数据·数据库·数据挖掘·数据分析·html
张哈大4 小时前
【 Redis | 实战篇 缓存 】
数据库·redis·笔记·spring·缓存