浅淡SqlServer的Top与Oracle的RowNum

浅淡SqlServer的Top与Oracle的RowNum,第1张

概述 平时的项目开发中,分页存储过程是用的比较多的存储过程,SqlServer分页存储过程中经常要用到top,Oracle中则经常用到了RowNum.        现在,有一个UserInfo表,一个字段是UserId,另一个字段是UserName,其中是UserId是自动增长的,步长是1.表中共有30条数据,其中UserId的值不一定是连续的。现在要实现的目的是取其中的第11至第20条记录。先看S

 平时的项目开发中,分页存储过程是用的比较多的存储过程,sqlServer分页存储过程中经常要用到top,Oracle中则经常用到了RowNum.

@H_301_9@        现在,有一个UserInfo表,一个字段是UserID,另一个字段是Username,其中是UserID是自动增长的,步长是1.表中共有30条数据,其中UserID的值不一定是连续的。现在要实现的目的是取其中的第11至第20条记录。先看sqlServer的几种做法:

@H_301_9@        第一种写法:

select   top   10 *   
 
from  UserInfo  
 
where  UserID  in  
       (
        
20  UserID 
        
 UserInfo
       ) 
 
order by  UserID  desc

 

@H_301_9@       第二种写法:

 UserInfo  not  

 (select top 10 UserID from UserInfo )

 

      第三种写法:

 UserID
>
 (   max (UserID)  top UserInfo order by UserID) a)

 

     第四种写法(只可在sqlserver 2005中):

 Row_Number() 
over (Order  UserID) as RowID , UserInfo) U 
where U.RowID between 10 and 20

 

     sqlserver 中其实还有另外几种写法,不一一写出。四种方法中,后两种的写法要比前两种写法效率要高些,但第四种只能写在sqlServer 2005中。

@H_301_9@     在看Oracle中实现取其中的第11至第20条记录的做法之前,先看看一些人在使用RowNum遇到的莫名其妙的怪事。表同样是UserInfo,30条数据

 t.  userinfo t   rownum > 10 @H_301_9@     查询结果:

@H_301_9@

@H_301_9@      

@H_301_9@     理论上应该是有20条数据才对啊,问题出现在哪呢?

@H_301_9@     因为ROWNUM是对结果集加的一个伪列,即先查到结果集之后再加上去的一个列 (这里要强调的一点是:先要有结果集)。简单的说 rownum 是对符合条件结果的序列号。所以对于rownum>10没有数据是否可以这样理解:

@H_301_9@     ROWNUM 是一个序列,是oracle数据库从数据文件或缓冲区中读取数据的顺序。它取得第一条记录则rownum值为1,第二条为2,依次类推。如果你用>,>=,=,between...and这些条件,因为从缓冲区或数据文件中得到的第一条记录的rownum为1,则被删除,接着取下条,可是它的rownum还是1,又被删除,依次类推,最后的查询结果为空。

@H_301_9@     再看下面一条SQL语句:

!=
 

@H_301_9@      查出的来结果不是21条,而是9条。可以这样理解:rownum 为9后的记录的 rownum为10,因条件为 !=10,所以去掉,其后记录补上,rownum又是10,也去掉,一直这样下去,最后的结果只有9条了。

@H_301_9@     如果把后面的条件改为 where rownum>1 时,会发现查不到一条数据,如果是where rownum>0 或是where rownum>=1时则可以查询到所有的数据。原因很简单:因为 rownum 是在查询到的结果集后加上去的,它总是从1开始。

@H_301_9@      between 1 and 20 或者 between 0 and 20 能查到结果,而用 between 2 and 20 却得不到结果,原因同上一样,因为 rownum 总是从 1 开始。

@H_301_9@      所以要实现取UserInfo表其中的第11至第20条记录,可以这样写:

  
 
(
 rownum  as  rn,t.  rownum  0 )
 rn  between and 20 @H_301_9@ 查询结果:

@H_301_9@  

     

@H_301_9@ 当然也可以这样写:

           UserInfo       rownum <      
   minus     
  
这种写法没有前面那种效率高。

@H_301_9@  

@H_301_9@ 但不能这样写:

select  t. *  from  UserInfo t  where  rownum  between  10  and  20
>  and <= 上 面两种写法都取不到数据的。 

@H_301_9@      对于sqlserver 和Oracle实现取表其中的第11至第20条记录如果有更好的写法 ,可以贴出来学习下。

总结

以上是内存溢出为你收集整理的浅淡SqlServer的Top与Oracle的RowNum全部内容,希望文章能够帮你解决浅淡SqlServer的Top与Oracle的RowNum所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/sjk/1177090.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-02
下一篇 2022-06-02

发表评论

登录后才能评论

评论列表(0条)

保存