oracle+先排序+再+rownum,关于oracle中rownum的使⽤探讨关于中rownum的使⽤探讨
⼀、rownum的使⽤
⽐对下⾯两个语句的异同
SELECT *
FROM (SELECT  a.*
FROM torderdetail a
ORDER BY order_date DESC)
WHERE ROWNUM <= 10
SELECT  a.*
FROM torderdetail a
WHERE ROWNUM <= 10
ORDER BY order_date DESC
之所以会出现这样的语句,主要是从效率上的考虑,前⾯条语句,是要进⾏全表扫描后再排序,然后再取10条纪录,后⼀条语句则不会全表扫描,只会取出10条纪录,很明显后条语句的效率会⾼许多。
oracle中trunc函数用法
那为什么会有争议呢,那就在于在执⾏顺序上争议,是先执⾏排序取10条纪录,还是取10条纪录,再排序呢?两种顺序取出来的结果是截然相反的,先排序再取10条,就是取最近的10条,⽽先取10条,再排序,则取出的最早的10条纪录。对于此语句,普遍的认为执⾏顺序是先取10条纪录再排序的。所以此语句应该是错误。
但实际上并⾮如此,此语句的执⾏顺序和order by的字段有关系,如果你order by 的字段是主建,则是先排序,再取10条(速度⽐第⼀种语句快),⽽排序字段不是主键时,是先取10条再排序,此时结果就与要求不⼀样了,所以第⼆种写法⼀定要在排序字段是主键的情况下才能保证结果正确。
⼆、rownum的扩展
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'))
三、
横向⽐较:SQL SERVER 2005中也引⽤了Row_number() over()这个分析函数,⽤法和oracle中是使⽤⼏乎完全⼀样

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。