数据库

位置:IT落伍者 >> 数据库 >> 浏览文章

Oracle查询rownum和rowid的区别


发布日期:2021年08月03日
 
Oracle查询rownum和rowid的区别

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

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

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

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

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

上一篇:oracle服务端和客户端之间的网络监听设置

下一篇:关于MySQL编码问题的经验总结