rowid是物理地址,用于定位oracle中具体数据的物理存储位置,而rownum则是sql的输出结果排序。
通俗的讲:rowid是相对不变的。rownum会变化,尤其是使用order by的时候。
因为rowid是不变的,所以努力求出rowid。然而,在虚拟表的rn, rd 被用作下一个步骤的输入
按rownum来分
1.rownum
关于 Oracle 的 rownum 问题,很多资料都说不支持>,>=,=,between...and,只能用以上符号(<、<=、!=)
参考https://docs.oracle.com/cd/B19306_01/server.102/b14200/pseudocolumns009.htm
Conditions testing for ROWNUM values greater than a positive integer are always false. For example, this query returns no rows:
SELECT * FROM employees
WHERE ROWNUM > 1;
The first row fetched is assigned a ROWNUM of 1 and makes the condition false. The second row to be fetched is now the first row
and is also assigned a ROWNUM of 1 and makes the condition false.
All rows subsequently fail to satisfy the condition, so no rows are returned.
数据源是from子句后的表,从数据源中先取出第一行数据,并标识rownum为1,根据WHERE ROWNUM > 1条件判断是不成立的,所以丢弃这一行数据,因为丢弃了所以结果集中是没有这一行记录存在的,那么rownum还是从1开始。接着又从数据源中取出第二行数据,并标识rownum为1(因为rownum是对取出的记录排序,虽然实际上 此一行记录是数据源中的第二行记录),根据WHERE ROWNUM > 1条件判断还是不成立的,以类类推。
如果从数据源中先取出第一行数据,并标识rownum为1,根据WHERE ROWNUM < 10 条件判断是成立的,接着又从数据源中取出第二行数据,并标识rownum为2,根据WHERE ROWNUM < 10条件判断是成立的,以类类推。
2.rowid
select t.*, t.rowid from EMP t
3.ROW_NUMBER() OVER函数
OVER()不能单独使用,要和分析函数:rank(), dense_rank(),
row_number() , sum() 等一起使用。
其参数:over(partition by columnname1 order by columnname2)
可实现按指定的字段分组排序,对于相同分组字段的结果集进行排序,
其中PARTITION BY 为分组字段,ORDER BY 指定排序字段
语法:
row_number() OVER (PARTITION BY COL1 ORDER BY COL2) 表示根据COL1分组,在分组内部根据 COL2排序,而此函数计算的值就表示每组内部排序后的顺序编号(组内连续的唯一的)
image.png
注意:部门20有2个3000的工资,排名却是不一样
image.png
4.排名函数
RANK ( ) OVER ( [query_partition_clause] order_by_clause )
DENSE_RANK ( ) OVER ( [query_partition_clause] order_by_clause )
可实现按指定的字段分组排序,对于相同分组字段的结果集进行排序,
其中PARTITION BY 为分组字段,ORDER BY 指定排序字段。
区别:
RANK() 有排名,但列名次的时候会产生不连续的排名编号,例如数据值 1,2,2,3 排名后发生的编号将是1,2,2,4;
DENSE_RANK() 有排名,但列名次的时候会产生连续的排名编号,例如数据值 1,2,2,3 排名后发生的编号将是1,2,2,3;
image.png image.png
网友评论