oracle 取前N条记录

本文介绍了Oracle数据库中实现分页查询及排序的方法,包括使用ROWNUM获取指定范围的记录和ROW_NUMBER() over()函数的应用。同时探讨了不同场景下查询语句的执行顺序和效率问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.oracle 取前10条记录

 1) select * from tbname where rownum < 11;

 2) select * from (select * from tbname order by id desc ) where rownum<=10;

 

下面是关于rownum的介绍
================================
三、 Rownum和row_number() over()的使用
ROWNUM是oracle从8开始提供的一个伪列,是把SQL出来的结果进行编号,始终从1开始,常见的用途就是用来分页输出.
比如 
SELECT *
  FROM torderdetail a
  WHERE ROWNUM <= 10
这条语句就是输出前10条纪录,在这里用途上类似于sql sever的top,不过rownum对于指定编号区间的输出应该说更强大
SELECT *
  FROM (SELECT a.*, ROWNUM rn
  FROM torderdetail a)
  WHERE rn >= 10 AND rn <= 20
这条语句即是输出第10到第20条纪录,这里之所以用rownum rn,是把rownum转成实例,因为rownum本身只能用<=的比较方式,只有转成实列,这样就可做 >=的比较了。
在实际用途中,常常会要求取最近的几条纪录,这就需要先对纪录进行排序后再取rownum<=
一般常见的
SELECT *
  FROM (SELECT a.*
  FROM torderdetail a
  ORDER BY order_date DESC)
 WHERE ROWNUM <= 10
而在优快云曾经发生过讨论,关于取近的10条纪录,有人给出这样的语句
SELECT a.*
  FROM torderdetail a
  WHERE ROWNUM <= 10
ORDER BY order_date DESC
之所以会出现这样的语句,主要是从效率上的考虑,前面条语句,是要进行全表扫描后再排序,然后再取10条纪录,后一条语句则不会全表扫描,只会取出10条纪录,很明显后条语句的效率会高许多。
那为什么会有争议呢,那就在于在执行顺序上争议,是先执行排序取10条纪录,还是取10条纪录,再排序呢?两种顺序取出来的结果是截然相反的,先排序再取10条,就是取最近的10条,而先取10条,再排序,则取出的最早的10条纪录。对于此语句,普遍的认为执行顺序是先取10条纪录再排序的。所以此语句应该是错误。但实际上并非如此,此语句的执行顺序和order by的字段有关系,如果你order by 的字段是pk,则是先排序,再取10条(速度比第一种语句快),而排序字段不是PK 时,是先取10条再排序,此时结果就与要求不一样了,所以第二种写法一定要在排序字段是主键的情况下才能保证结果正确。
Row_number() over()这个分析函数是从9I开始提供的,一般的用途和rownum差不多。
一般写法row_number() over( order by order_date desc) 生成的顺序和rownum的语句一样,效率也一样(对于同样有order by 的rownum语句来说),所以在这种情况下两种用法是一样的。
而对于分组后取最近的10条纪录,则是rownum无法实现的,这时只有row_number可以实现,row_number() over(partition by 分组字段 order by 排序字段)就能实现分组后编号,比如说要取近一个月的每天最后10个订单纪录
SELECT *
  FROM (SELECT a.*,
ROW_NUMBER () OVER (PARTITION BY TRUNC (order_date) ORDER BY order_date DESC)
  rn
  FROM torderdetail a)
 WHERE rn <= 10
Rownum的另类用法,有时候我们会遇到这种需求,要求输出当月的所有天数,许多人会烦恼,数据库里又没有这样的表,怎么输出一个月的所有天数呢?用rownum就能解决:
SELECT TRUNC (SYSDATE, 'MM') + ROWNUM - 1
  FROM DUAL
CONNECT BY ROWNUM <= TO_NUMBER (TO_CHAR (LAST_DAY (SYSDATE), 'dd'))

<think>我们之已经讨论过在Oracle 10g中2数据的方法,主要使用ROWNUM。这里再详细说明一下。 在Oracle 10g中,没有像MySQL的LIMIT或Oracle 12c的FETCH FIRST这样的语法,所以通常使用ROWNUM来实现。 注意:ROWNUM是在查询结果返回之分配的,因此不能直接在WHERE子句中使用ROWNUM>某个数(比如第3到5),但可以用子查询来实现分页。 现在,针对“10数据中的2”,我们可以理解为2,或者任意2?但通常我们是指2(按某种顺序或默认顺序)。 如果不需要排序,直接2,那么: SELECT * FROM table_name WHERE ROWNUM <= 2; 但是,如果需要按某个字段排序后2,那么就需要在子查询中先排序,再使用ROWNUM。 下面给出两种情况的代码和解释:</think>### Oracle 10g 中 2 数据的解决方案 在 Oracle 10g 中,可以使用 `ROWNUM` 伪列或 `ROW_NUMBER()` 分析函数实现只 2 数据。以下是两种方法的详细实现: #### 方法 1:使用 `ROWNUM`(推荐) ```sql SELECT * FROM ( SELECT t.* FROM your_table t -- 替换 your_table 为实际表名 WHERE ROWNUM <= 2 -- 直接限制行数 ); ``` #### 方法 2:使用 `ROW_NUMBER()`(带排序) ```sql SELECT * FROM ( SELECT t.*, ROW_NUMBER() OVER (ORDER BY your_column) AS rn -- 替换 your_column 为排序列 FROM your_table t ) WHERE rn <= 2; ``` ### 代码解释 1. **`ROWNUM` 方法**: - `ROWNUM` 是 Oracle 自动为查询结果分配的行号(从 1 开始) - 直接在 `WHERE` 子句中使用 `ROWNUM <= 2` 过滤 2 行 - **优点**:简单高效,不需要子查询嵌套 - **限制**:无法直接指定排序顺序 2. **`ROW_NUMBER()` 方法**: - `ROW_NUMBER() OVER (ORDER BY ...)` 为排序后的结果分配序号 - 外层查询通过 `WHERE rn <= 2` 过滤 2 行 - **优点**:支持自定义排序规则 - **适用场景**:需要按特定顺序(如时间倒序) N 记录 ### 示例场景 假设有表 `employees`,获 2 记录: ```sql -- 方法 1:直接 2 SELECT employee_id, first_name, last_name FROM employees WHERE ROWNUM <= 2; -- 方法 2:按工资降序 2 名 SELECT * FROM ( SELECT employee_id, first_name, salary, ROW_NUMBER() OVER (ORDER BY salary DESC) AS rnk FROM employees ) WHERE rnk <= 2; ``` ### 注意事项 1. `ROWNUM` 是在数据**返回**分配的,件 `ROWNUM = 2` 无效(需用 `<=`) 2. 分页查询需结合子查询(如第 3-5 ): ```sql SELECT * FROM ( SELECT t.*, ROWNUM AS rn FROM your_table t WHERE ROWNUM <= 5 -- 上限 ) WHERE rn >= 3; -- 下限 ``` 3. Oracle 12c+ 可使用更简洁的 `FETCH FIRST 2 ROWS ONLY` 语法 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值