rownum和rowid的区别(转)

在Oracle 中,有一个很有趣的东西,那就是rownum 。当你从某个表中查询数据的时候,返回的结果集中都会带有rownum 这个字段,而且有时候也可以使用rownum 进行一些条件查询。

在查询中,我们可以注意到,类似于“select xx from table where rownum < n ”(n>1 )这样的查询是有正确含义的,而“select xx from table where rownum = n ”这样的查询只在n =1 的时候成立,“select xx from table where rownum > n ”(n>1 )这样的查询只能得到一个空集。另外“select xx from table where rownum > 0 ”这个查询会返回所有的记录。这是为什么呢?原因就在于Oracle 对rownum 的处理上,rownum 是在得到结果集的时候产生的,用于标记结果集中结果顺序的一个字段,这个字段被称为“伪数列”,也就是事实上不存在的一个数列。它的特点是按顺序标记,而且是逐次递加的,换句话说就是只有有rownum =1 的记录,才可能有rownum =2 的记录。

让我们回头来分析一下在where 中使用rownum 作为查询条件的情况。在rownum 取=1 ,或者rownum <= n (n>1) 的时候,没有问题。那么为什么当条件为rownum = n 或者rownum >= n 时明明有数据却只能得到一个空集呢?假设我们的查询条件为rownum = 2 ,那么在查询出的第一条记录的时候,oracle 标记此条记录rownum 为1 ,结果发现和rownum=2 的条件不符,于是结果集为空。写到这里,我忽然有一个有趣的想法:假如有一条查询语句为select xx,yy from table where zz > 20 and rownum < 10 ,那么在执行的时候,是先按照zz>20 的条件查询出一个结果集,然后按照rownum 取出前10 条返回?还是在按照zz>20 的条件先查询,然后有一个记录就标记一个rownum ,到rownum<10 的时候就停止查询?我觉得应该是后者,也就是在执行语句的时候,不是做full scan ,而是取够数据就停止查询。要验证这个想法应该很简单,找一个数据量非常大的表进行查询就可以了。可惜目前我没有这样的表。

我们可以看出,直接使用rownum 是要受到限制的。但是很容易遇到这样的需求“查出符合条件的第xx 条到第xx 条记录”,比如页面的分页处理。这个时候如何构造出适合自己的结果集?嗯,墙边那位说全取出来手工挑选的哥们可以拉出去了。当然这样做也是可以的,但是前提是整个数据集的数据条数不多的情况下。假如遇到上十万百条的数据,全部取出来的话,用户就不用干别的事情了。这个时候用户应该怎么做呢?当然就是要用到我们介绍的rownum 拉!rownum 不是个“伪数列”么,好说,我们现在把它弄成一个实在的字段就可以了。具体做法就是利用子查询,在构建临时表的时候,把rownum 也一起构造进去。比如“select xx,yy from (select xx,yy,rownum as xyz from table where zz >20) where xyz between 10 and 20 ”这样就可以了。另外使用oracle 提供的结果集处理函数minus 也可以做到,例如“select xx,yy from table where zz > 20 and rownum <20 minus select xx,yy from table where zz>20 and rownum <10 ”,但是使用minus 好像比使用子查询更加消耗资源。

和rownum 相似,oracle 还提供了另外一个伪数列:rowid 。不过rowid 和rownum 不同,一般说来每一行数据对应的rowid 是固定而且唯一的,在这一行数据存入数据库 的时候就确定了。可以利用rowid 来查询记录,而且通过rowid 查询记录是查询速度最快的查询方法。(这个我没有试过,另外要记住一个长度在18 位,而且没有太明显规律的字符串是一个很困难的事情,所以我个人认为利用rowid 查询记录的实用性不是很大)rowid 只有在表发生移动(比如表空间变化,数据导入/ 导出以后),才会发生变化。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值