浅淡SqlServer的Top与Oracle的RowNum 转载自http://www.cnblogs.com/liguangxi8/archive/2010/05/21/1740708.html
2012-09-20 09:10
756 查看
平时的项目开发中,分页存储过程是用的比较多的存储过程,SqlServer分页存储过程中经常要用到top,Oracle中则经常用到了RowNum.
现在,有一个UserInfo表,一个字段是UserId,另一个字段是UserName,其中是UserId是自动增长的,步长是1.表中共有30条数据,其中UserId的值不一定是连续的。现在要实现的目的是取其中的第11至第20条记录。先看SqlServer的几种做法:
第一种写法:
select top 10 *
from UserInfo
where UserId in
(
select top 20 UserId
from UserInfo
)
order by UserId desc
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E4%B8%80%E7%A7%8D.jpg)
第二种写法:
select top 10 * from UserInfo where UserId not in
(select top 10 UserId from UserInfo )
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E4%BA%8C%E7%A7%8D.jpg)
第三种写法:
select top 10 * from UserInfo where UserId>
(select max(UserId) from
(select top10 UserId from UserInfo order by UserId) a)
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E4%BA%8C%E7%A7%8D.jpg)
第四种写法(只可在Sqlserver 2005中):
select * from (select Row_Number() over
(Order by UserId) as RowId ,* from UserInfo) U
where U.RowId between 10 and 20
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E5%9B%9B%E7%A7%8D.jpg)
Sqlserver 中其实还有另外几种写法,不一一写出。四种方法中,后两种的写法要比前两种写法效率要高些,但第四种只能写在SqlServer 2005中。
在看Oracle中实现取其中的第11至第20条记录的做法之前,先看看一些人在使用RowNum遇到的莫名其妙的怪事。表同样是UserInfo,30条数据
select t.*from userinfo t where rownum>10
查询结果:
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/Oracle02.jpg)
理论上应该是有20条数据才对啊,问题出现在哪呢?
因为ROWNUM是对结果集加的一个伪列,即先查到结果集之后再加上去的一个列 (这里要强调的一点是:先要有结果集)。简单的说 rownum 是对符合条件结果的序列号。所以对于rownum>10没有数据是否可以这样理解:
ROWNUM 是一个序列,是oracle数据库从数据文件或缓冲区中读取数据的顺序。它取得第一条记录则rownum值为1,第二条为2,依次类推。如果你用>,>=,=,between...and这些条件,因为从缓冲区或数据文件中得到的第一条记录的rownum为1,则被删除,接着取下条,可是它的rownum还是1,又被删除,依次类推,最后的查询结果为空。
再看下面一条sql语句:
select t.* from userinfo t where rownum!=10
查询结果:
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/Oracle03.jpg)
查出的来结果不是21条,而是9条。可以这样理解:rownum 为9后的记录的 rownum为10,因条件为 !=10,所以去掉,其后记录补上,rownum又是10,也去掉,一直这样下去,最后的结果只有9条了。
如果把后面的条件改为 where rownum>1 时,会发现查不到一条数据,如果是where rownum>0 或是where rownum>=1时则可以查询到所有的数据。原因很简单:因为 rownum 是在查询到的结果集后加上去的,它总是从1开始。
between 1 and 20 或者 between 0 and 20 能查到结果,而用 between 2 and 20 却得不到结果,原因同上一样,因为 rownum 总是从 1 开始。
所以要实现取UserInfo表其中的第11至第20条记录,可以这样写:
select * from
(select rownum as rn,t.* from userinfo t where rownum >0)
where rn between 10 and 20
查询结果:
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/Oracle04.jpg)
当然也可以这样写:
select * from UserInfo where rownum<20
minus
select * from UserInfo where rownum<10
这种写法没有前面那种效率高。
但不能这样写:
select t.* from UserInfo t where rownum between 10 and 20
select t.* from UserInfo t where rownum > 10 and rownum <=20
上 面两种写法都取不到数据的。
对于Sqlserver 和Oracle实现取表其中的第11至第20条记录如果有更好的写法 ,可以贴出来学习下。
现在,有一个UserInfo表,一个字段是UserId,另一个字段是UserName,其中是UserId是自动增长的,步长是1.表中共有30条数据,其中UserId的值不一定是连续的。现在要实现的目的是取其中的第11至第20条记录。先看SqlServer的几种做法:
第一种写法:
select top 10 *
from UserInfo
where UserId in
(
select top 20 UserId
from UserInfo
)
order by UserId desc
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E4%B8%80%E7%A7%8D.jpg)
第二种写法:
select top 10 * from UserInfo where UserId not in
(select top 10 UserId from UserInfo )
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E4%BA%8C%E7%A7%8D.jpg)
第三种写法:
select top 10 * from UserInfo where UserId>
(select max(UserId) from
(select top10 UserId from UserInfo order by UserId) a)
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E4%BA%8C%E7%A7%8D.jpg)
第四种写法(只可在Sqlserver 2005中):
select * from (select Row_Number() over
(Order by UserId) as RowId ,* from UserInfo) U
where U.RowId between 10 and 20
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/SS%E7%AC%AC%E5%9B%9B%E7%A7%8D.jpg)
Sqlserver 中其实还有另外几种写法,不一一写出。四种方法中,后两种的写法要比前两种写法效率要高些,但第四种只能写在SqlServer 2005中。
在看Oracle中实现取其中的第11至第20条记录的做法之前,先看看一些人在使用RowNum遇到的莫名其妙的怪事。表同样是UserInfo,30条数据
select t.*from userinfo t where rownum>10
查询结果:
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/Oracle02.jpg)
理论上应该是有20条数据才对啊,问题出现在哪呢?
因为ROWNUM是对结果集加的一个伪列,即先查到结果集之后再加上去的一个列 (这里要强调的一点是:先要有结果集)。简单的说 rownum 是对符合条件结果的序列号。所以对于rownum>10没有数据是否可以这样理解:
ROWNUM 是一个序列,是oracle数据库从数据文件或缓冲区中读取数据的顺序。它取得第一条记录则rownum值为1,第二条为2,依次类推。如果你用>,>=,=,between...and这些条件,因为从缓冲区或数据文件中得到的第一条记录的rownum为1,则被删除,接着取下条,可是它的rownum还是1,又被删除,依次类推,最后的查询结果为空。
再看下面一条sql语句:
select t.* from userinfo t where rownum!=10
查询结果:
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/Oracle03.jpg)
查出的来结果不是21条,而是9条。可以这样理解:rownum 为9后的记录的 rownum为10,因条件为 !=10,所以去掉,其后记录补上,rownum又是10,也去掉,一直这样下去,最后的结果只有9条了。
如果把后面的条件改为 where rownum>1 时,会发现查不到一条数据,如果是where rownum>0 或是where rownum>=1时则可以查询到所有的数据。原因很简单:因为 rownum 是在查询到的结果集后加上去的,它总是从1开始。
between 1 and 20 或者 between 0 and 20 能查到结果,而用 between 2 and 20 却得不到结果,原因同上一样,因为 rownum 总是从 1 开始。
所以要实现取UserInfo表其中的第11至第20条记录,可以这样写:
select * from
(select rownum as rn,t.* from userinfo t where rownum >0)
where rn between 10 and 20
查询结果:
![](http://images.cnblogs.com/cnblogs_com/liguangxi8/Oracle04.jpg)
当然也可以这样写:
select * from UserInfo where rownum<20
minus
select * from UserInfo where rownum<10
这种写法没有前面那种效率高。
但不能这样写:
select t.* from UserInfo t where rownum between 10 and 20
select t.* from UserInfo t where rownum > 10 and rownum <=20
上 面两种写法都取不到数据的。
对于Sqlserver 和Oracle实现取表其中的第11至第20条记录如果有更好的写法 ,可以贴出来学习下。
相关文章推荐
- [转载]IIS 6.0配置HTTP压缩的步骤 http://www.cnblogs.com/dudu/archive/2004/12/12/76085.html
- 转载String分析 (http://www.cnblogs.com/fancydeepin/archive/2013/04/22/min-snail-speak_String.html)
- 有关T-SQL的10个好习惯 转载http://www.cnblogs.com/CareySon/archive/2012/10/11/2719598.html
- PowerDesigner教程系列(五)概念数据模型 转载(http://www.cnblogs.com/yxonline/archive/2007/04/09/705479.html)
- 转载:C++设计模式之Bridge模式 http://www.cnblogs.com/chain2012/archive/2011/01/17/1937528.html
- 转载自:http://www.cnblogs.com/chenyadong/archive/2011/10/25/2223610.html
- <转载>Bootstrap 入门教程 http://www.cnblogs.com/ventlam/archive/2012/05/28/2520703.html 系列
- MySQL 请选择合适的列! 转载(http://www.cnblogs.com/baochuan/archive/2012/05/23/2513224.html)
- JavaScript逻辑操作符 转载自博客园 http://www.cnblogs.com/pigtail/archive/2012/03/09/2387486.html
- HBase Coprocessor 剖析与编程实践(转载http://www.cnblogs.com/ventlam/archive/2012/10/30/2747024.html)
- 转载至http://www.cnblogs.com/huxi/archive/2010/07/04/1771073.html
- 使用Sqlserver事务发布实现数据同步(引:http://www.cnblogs.com/daizhj/archive/2009/11/18/1605293.html )
- SqlServer 监控数据库连接数方法(转自:http://www.cnblogs.com/alphacn/archive/2010/07/19/1780572.html)
- 事件转载自(http://www.cnblogs.com/zengshunyou/archive/2009/06/16/1504475.html)
- linux的硬链接与软连接(转载,原文地址为http://www.cnblogs.com/sonic4x/archive/2011/08/05/2128543.html)
- 转载TortoiseSVN的使用详解1(http://www.cnblogs.com/xiaobaihome/archive/2012/03/20/2407610.html)
- C#正则表达式整理备忘 转载(http://www.cnblogs.com/KissKnife/archive/2008/03/23/1118423.html)
- WPF企业内训全程实录(上) 转载 http://www.cnblogs.com/KnightsWarrior/archive/2010/10/28/1863249.html
- C#中Dictionary的用法及用途 原文转载自:http://www.cnblogs.com/linzheng/archive/2010/12/13/1904709.html
- 转载:.net 操作FTP from http://www.cnblogs.com/aierong/archive/2008/12/08/ftp.html